summaryrefslogtreecommitdiff
path: root/plugin.video.mmlive/resources
diff options
context:
space:
mode:
authoreracknaphobia <eracknaphobia@hotmail.com>2017-03-19 11:28:22 -0400
committerenen92 <enen92@users.noreply.github.com>2017-03-19 15:28:22 +0000
commit4802b7e3b5cea3b306712e756226ce37b3e743e9 (patch)
tree48fe4369fd6c7316d465f4a1beacb93234e2ef2c /plugin.video.mmlive/resources
parent07733ec5215ea46855eee4c648c582e1e9ae9d51 (diff)
[plugin.video.mmlive] 2017.3.19 (#1063)
* [plugin.video.mmlive] 2017.3.18 * Fixes for Krypton Guidelines
Diffstat (limited to 'plugin.video.mmlive/resources')
-rw-r--r--plugin.video.mmlive/resources/__init__.py1
-rw-r--r--plugin.video.mmlive/resources/adobepass.py224
-rw-r--r--plugin.video.mmlive/resources/globals.py437
-rw-r--r--plugin.video.mmlive/resources/language/resource.language.en_gb/strings.po41
-rw-r--r--plugin.video.mmlive/resources/settings.xml7
5 files changed, 710 insertions, 0 deletions
diff --git a/plugin.video.mmlive/resources/__init__.py b/plugin.video.mmlive/resources/__init__.py
new file mode 100644
index 0000000..2b620f6
--- /dev/null
+++ b/plugin.video.mmlive/resources/__init__.py
@@ -0,0 +1 @@
+# dummy file to init directory \ No newline at end of file
diff --git a/plugin.video.mmlive/resources/adobepass.py b/plugin.video.mmlive/resources/adobepass.py
new file mode 100644
index 0000000..e08be36
--- /dev/null
+++ b/plugin.video.mmlive/resources/adobepass.py
@@ -0,0 +1,224 @@
+import os, sys
+import uuid, hmac, hashlib, base64, time
+import xbmc, xbmcgui, xbmcaddon
+import cookielib, urllib, urllib2, json
+from urllib2 import URLError, HTTPError
+
+class ADOBE():
+ api_url = 'http://api.auth.adobe.com'
+ base_url = 'http://sp.auth.adobe.com'
+ activate_url = ''
+ requestor_id = ''
+ public_key = ''
+ private_key = ''
+ device_id = ''
+ regcode = ''
+ user_agent = 'Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/43.0.2357.81 Safari/537.36'
+
+
+ def __init__(self, service_vars):
+ self.requestor_id = service_vars['requestor_id']
+ self.public_key = service_vars['public_key']
+ self.private_key = service_vars['private_key']
+ self.activate_url = service_vars['activate_url']
+ self.device_id = self.getDeviceID()
+
+
+ def getDeviceID(self):
+ addon_profile_path = xbmc.translatePath(xbmcaddon.Addon().getAddonInfo('profile'))
+ fname = os.path.join(addon_profile_path, 'device.id')
+ #xbmc.log("FILE PATH == "+str(fname))
+ if not os.path.isfile(fname):
+ if not os.path.exists(addon_profile_path):
+ os.makedirs(addon_profile_path)
+ new_device_id =str(uuid.uuid1())
+ device_file = open(fname,'w')
+ device_file.write(new_device_id)
+ device_file.close()
+
+ fname = os.path.join(addon_profile_path, 'device.id')
+ device_file = open(fname,'r')
+ device_id = device_file.readline()
+ device_file.close()
+
+ return device_id
+
+
+ def createAuthorization(self, request_method, request_uri):
+ nonce = str(uuid.uuid4())
+ epochtime = str(int(time.time() * 1000))
+ authorization = request_method + " requestor_id="+self.requestor_id+", nonce="+nonce+", signature_method=HMAC-SHA1, request_time="+epochtime+", request_uri="+request_uri
+ signature = hmac.new(self.private_key , authorization, hashlib.sha1)
+ signature = base64.b64encode(signature.digest())
+ authorization += ", public_key="+self.public_key+", signature="+signature
+
+ return authorization
+
+
+ def registerDevice(self):
+ reggie_url = '/reggie/v1/'+self.requestor_id+'/regcode'
+ authorization = self.createAuthorization('POST',reggie_url)
+ url = self.api_url+reggie_url
+ headers = [ ("Accept", "*/*"),
+ ("Content-type", "application/x-www-form-urlencoded"),
+ ("Authorization", authorization),
+ ("Accept-Language", "en-US"),
+ ("Accept-Encoding", "gzip, deflate"),
+ ("User-Agent", self.user_agent),
+ ("Connection", "Keep-Alive"),
+ ("Pragma", "no-cache")
+ ]
+
+
+ body = 'registrationURL='+self.base_url+'/adobe-services'
+ body += '&ttl=2700'
+ body += '&deviceId='+self.device_id
+ body += '&format=json'
+
+ json_source = self.requestJSON(url, headers, body)
+
+
+ msg = '1. Go to [B][COLOR yellow]'+self.activate_url+'[/COLOR][/B][CR]'
+ msg += '2. Select any platform, it does not matter[CR]'
+ msg += '3. Enter [B][COLOR yellow]'+json_source['code']+'[/COLOR][/B] as your activation code'
+ self.regcode = json_source['code']
+ dialog = xbmcgui.Dialog()
+ ok = dialog.ok('Activate Device', msg)
+
+
+
+ def authorizeDevice(self, resource_id):
+ auth_url = '/api/v1/authorize'
+ authorization = self.createAuthorization('GET',auth_url)
+ url = self.api_url+auth_url
+ url += '?deviceId='+self.device_id
+ url += '&requestor='+self.requestor_id
+ url += '&resource='+urllib.quote(resource_id)
+
+ url += '&format=json'
+ #req = urllib2.Request(url)
+
+ headers = [ ("Accept", "*/*"),
+ ("Content-type", "application/x-www-form-urlencoded"),
+ ("Authorization", authorization),
+ ("Accept-Language", "en-US"),
+ ("Accept-Encoding", "deflate"),
+ ("User-Agent", self.user_agent),
+ ("Connection", "Keep-Alive"),
+ ("Pragma", "no-cache")
+ ]
+
+ json_source = self.requestJSON(url, headers)
+ mvpd = json_source['mvpd']
+
+ return mvpd
+
+
+ def authenticate(self):
+ auth_url = '/api/v1/authenticate/PL2MVJL'
+ authorization = self.createAuthorization('GET',auth_url)
+ url = self.api_url+auth_url
+ url += '?deviceId='+self.device_id
+ url += '&requestor='+self.requestor_id
+ url += '&deviceType=Win8Universal'
+ #req = urllib2.Request(url)
+
+ headers = [ ("Accept", "*/*"),
+ ("Content-type", "application/x-www-form-urlencoded"),
+ ("Authorization", authorization),
+ ("Accept-Language", "en-US"),
+ ("Accept-Encoding", "deflate"),
+ ("User-Agent", self.user_agent),
+ ("Connection", "Keep-Alive"),
+ ("Pragma", "no-cache")
+ ]
+
+ json_source = self.requestJSON(url, headers)
+
+ def deauthorizeDevice(self):
+ auth_url = '/api/v1/logout'
+ authorization = self.createAuthorization('DELETE',auth_url)
+ url = self.api_url+auth_url
+ url += '?deviceId='+self.device_id
+ url += '&requestor='+self.requestor_id
+ url += '&format=json'
+ #req = urllib2.Request(url)
+
+ headers = [ ("Accept", "*/*"),
+ ("Content-type", "application/x-www-form-urlencoded"),
+ ("Authorization", authorization),
+ ("Accept-Language", "en-US"),
+ ("Accept-Encoding", "deflate"),
+ ("User-Agent", self.user_agent),
+ ("Connection", "Keep-Alive"),
+ ("Pragma", "no-cache")
+ ]
+
+ try: json_source = self.requestJSON(url, headers, None, 'DELETE')
+ except: pass
+
+
+ def mediaToken(self, resource_id):
+ url = 'http://api.auth.adobe.com/api/v1/tokens/media'
+ url += '?deviceId='+self.device_id
+ url += '&requestor='+self.requestor_id
+ url += '&resource='+urllib.quote(resource_id)
+ url += '&format=json'
+ authorization = self.createAuthorization('GET','/api/v1/tokens/media')
+ headers = [ ("Accept", "*/*"),
+ ("Content-type", "application/x-www-form-urlencoded"),
+ ("Authorization", authorization),
+ ("Accept-Language", "en-US"),
+ ("Accept-Encoding", "deflate"),
+ ("User-Agent", self.user_agent),
+ ("Connection", "Keep-Alive"),
+ ("Pragma", "no-cache")
+ ]
+
+ json_source = self.requestJSON(url, headers)
+
+ return json_source['serializedToken']
+
+
+
+ def requestJSON(self, url, headers, body=None, method=None):
+ addon_profile_path = xbmc.translatePath(xbmcaddon.Addon().getAddonInfo('profile'))
+ cj = cookielib.LWPCookieJar(os.path.join(addon_profile_path, 'cookies.lwp'))
+ try: cj.load(os.path.join(addon_profile_path, 'cookies.lwp'),ignore_discard=True)
+ except: pass
+ opener = urllib2.build_opener(urllib2.HTTPCookieProcessor(cj))
+ opener.addheaders = headers
+ json_source = ''
+
+ try:
+ request = urllib2.Request(url, body)
+ if method == 'DELETE': request.get_method = lambda: method
+ response = opener.open(request)
+ json_source = json.load(response)
+ response.close()
+ self.saveCookie(cj)
+ except HTTPError as e:
+ if e.code == 403:
+ msg = 'Your device is not authorized to view the selected stream.\n Would you like to authorize this device now?'
+ dialog = xbmcgui.Dialog()
+ answer = dialog.yesno('Account Not Authorized', msg)
+ if answer:
+ self.registerDevice()
+ else:
+ sys.exit(0)
+ else:
+ sys.exit(0)
+
+ return json_source
+
+
+ def saveCookie(self, cj):
+ # Cookielib patch for Year 2038 problem
+ # Possibly wrap this in if to check if device is using a 32bit OS
+ for cookie in cj:
+ # Jan, 1 2038
+ if cookie.expires >= 2145916800:
+ #Jan, 1 2037
+ cookie.expires = 2114380800
+
+ cj.save(ignore_discard=True) \ No newline at end of file
diff --git a/plugin.video.mmlive/resources/globals.py b/plugin.video.mmlive/resources/globals.py
new file mode 100644
index 0000000..cc5fd2a
--- /dev/null
+++ b/plugin.video.mmlive/resources/globals.py
@@ -0,0 +1,437 @@
+import uuid
+import hmac
+import hashlib
+import string, random
+from StringIO import StringIO
+import gzip
+from urllib2 import URLError, HTTPError
+import sys
+import xbmc,xbmcplugin, xbmcgui, xbmcaddon
+import re, os, time
+import urllib, urllib2
+import json
+import HTMLParser
+import calendar
+from datetime import datetime, timedelta
+import time
+import cookielib
+import base64
+
+
+addon_handle = int(sys.argv[1])
+SCORE_COLOR = 'FF00B7EB'
+UPCOMING = 'FFD2D2D2'
+CRITICAL ='FFD10D0D'
+FINAL = 'FF666666'
+FREE = 'FF43CD80'
+GAMETIME_COLOR = 'FFFFFF66'
+now = datetime.now()
+BASE_PATH = 'https://data.ncaa.com/mml/'+str(now.year)+'/mobile'
+
+def colorString(string, color):
+ return '[COLOR='+color+']'+string+'[/COLOR]'
+
+def stringToDate(string, date_format):
+ try:
+ date = datetime.strptime(str(string), date_format)
+ except TypeError:
+ date = datetime(*(time.strptime(str(string), date_format)[0:6]))
+
+ return date
+
+def FIND(source,start_str,end_str):
+ start = source.find(start_str)
+ end = source.find(end_str,start+len(start_str))
+
+ if start != -1:
+ return source[start+len(start_str):end]
+ else:
+ return ''
+
+
+def SET_STREAM_QUALITY(url):
+
+ stream_url = {}
+ stream_title = []
+
+ #Open master file a get cookie(s)
+ cj = cookielib.LWPCookieJar()
+ opener = urllib2.build_opener(urllib2.HTTPCookieProcessor(cj))
+ opener.addheaders = [ ("Accept", "*/*"),
+ ("Accept-Encoding", "deflate"),
+ ("Accept-Language", "en-us"),
+ ("User-Agent", UA_MMOD)]
+
+ resp = opener.open(url)
+ master = resp.read()
+ resp.close()
+
+ cookies = ''
+ for cookie in cj:
+ if cookies != '':
+ cookies = cookies + "; "
+ cookies = cookies + cookie.name + "=" + cookie.value
+
+ line = re.compile("(.+?)\n").findall(master)
+
+ for temp_url in line:
+ if '#EXT' not in temp_url:
+ temp_url = temp_url.rstrip()
+ start = 0
+ if 'http' not in temp_url:
+ if 'master' in url:
+ start = url.find('master')
+ elif 'manifest' in url:
+ start = url.find('manifest')
+
+ if url.find('?') != -1:
+ replace_url_chunk = url[start:url.find('?')]
+ else:
+ replace_url_chunk = url[start:]
+
+
+ temp_url = url.replace(replace_url_chunk,temp_url)
+ temp_url = temp_url.rstrip() + "|User-Agent=" + UA_MMOD
+
+ #if cookies != '':
+ #temp_url = temp_url + "&Cookie=" + cookies
+
+ stream_title.append(desc)
+ stream_url.update({desc:temp_url})
+ else:
+ desc = ''
+ start = temp_url.find('BANDWIDTH=')
+ if start > 0:
+ start = start + len('BANDWIDTH=')
+ end = temp_url.find(',',start)
+ desc = temp_url[start:end]
+ try:
+ int(desc)
+ desc = str(int(desc)/1000) + ' kbps'
+ except:
+ pass
+
+
+ if len(stream_title) > 0:
+ ret =-1
+ stream_title.sort(key=natural_sort_key)
+ print "PLAY BEST SETTING"
+ print PLAY_BEST
+ if str(PLAY_BEST) == 'true':
+ ret = len(stream_title)-1
+ else:
+ dialog = xbmcgui.Dialog()
+ ret = dialog.select('Choose Stream Quality', stream_title)
+ print ret
+ if ret >=0:
+ url = stream_url.get(stream_title[ret])
+ else:
+ sys.exit()
+ else:
+ msg = "No playable streams found."
+ dialog = xbmcgui.Dialog()
+ ok = dialog.ok('Streams Not Found', msg)
+
+
+ return url
+
+
+def natural_sort_key(s):
+ _nsre = re.compile('([0-9]+)')
+ return [int(text) if text.isdigit() else text.lower()
+ for text in re.split(_nsre, s)]
+
+
+def SAVE_COOKIE(cj):
+ # Cookielib patch for Year 2038 problem
+ # Possibly wrap this in if to check if device is using a 32bit OS
+ for cookie in cj:
+ # Jan, 1 2038
+ if cookie.expires >= 2145916800:
+ #Jan, 1 2037
+ cookie.expires = 2114380800
+
+ cj.save(ignore_discard=True);
+
+
+
+
+def getTournamentInfo():
+ now = datetime.now()
+ url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/tournament.json'
+ req = urllib2.Request(url)
+ #req.add_header('User-Agent', 'Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/41.0.2228.0 Safari/537.36')
+ req.add_header('Connection', 'keep-alive')
+ req.add_header('Accept', '*/*')
+ req.add_header('User-Agent', UA_MMOD)
+ req.add_header('Accept-Language', 'en-us')
+ req.add_header('Accept-Encoding', 'deflate')
+
+ response = urllib2.urlopen(req)
+ json_source = json.load(response)
+ response.close()
+
+ teams = json.dumps(json_source['tournament']['teams']['team'])
+
+ return teams
+
+def getTeamInfo(teams, team_id):
+ all_teams = json.loads(teams)
+ team_name = ''
+ link_name = ''
+
+ for team in all_teams:
+ if str(team['id']) == str(team_id):
+ #team_name = team['school']
+ #link_name = team['link']
+ break
+
+ return team
+
+def getCurrentInfo():
+ now = datetime.now()
+ url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/current.json'
+ req = urllib2.Request(url)
+ #req.add_header('User-Agent', 'Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/41.0.2228.0 Safari/537.36')
+ req.add_header('Connection', 'keep-alive')
+ req.add_header('Accept', '*/*')
+ req.add_header('User-Agent', UA_MMOD)
+ req.add_header('Accept-Language', 'en-us')
+ req.add_header('Accept-Encoding', 'deflate')
+
+ response = urllib2.urlopen(req)
+ json_source = json.load(response)
+ response.close()
+
+ current_games = ''
+
+ try:
+ current_games = json.dumps(json_source['current']['game'])
+ except:
+ pass
+
+ return current_games
+
+
+def getGameClock(current_games, game_id):
+ games = json.loads(current_games)
+ clock = 'Final'
+ ordinal_indicator = ''
+
+ for game in games:
+ if str(game['id']) == str(game_id):
+ print game
+ clock = str(game['clock'])
+ per = str(game['per'])
+ state = str(game['state'])
+ if state != '4' and per != '':
+ if per == '1' and clock == '00:00':
+ clock = 'HALF'
+ else:
+ if per == '1':
+ ordinal_indicator = "st"
+ elif per == '2':
+ ordinal_indicator = "nd"
+ clock = clock + ' ' + per + ordinal_indicator
+ break
+
+
+ return clock
+
+
+def getAppConfig():
+ #https://data.ncaa.com/mml/2017/mobile/appConfig_iPhone.json
+ now = datetime.now()
+ url = 'https://data.ncaa.com/mml/'+str(now.year)+'/mobile/appConfig_iPhone.json'
+ req = urllib2.Request(url)
+ req.add_header('Accept', '*/*')
+ req.add_header('User-Agent', UA_IPHONE)
+ req.add_header('Accept-Language', 'en-us')
+ req.add_header('Accept-Encoding', 'deflate')
+
+ response = urllib2.urlopen(req)
+ json_source = json.load(response)
+ response.close()
+
+ api = json_source['api']
+ #BASE_PATH = api['base']['sche']
+
+
+
+def tokenTurner(media_token, stream_url, mvpd):
+ #url = 'http://token.vgtf.net/token/turner'
+ url = 'https://token.vgtf.net/token/token_spe_mml?profile=mml'
+
+ cj = cookielib.LWPCookieJar()
+ cj.load(os.path.join(ADDON_PATH_PROFILE, 'cookies.lwp'),ignore_discard=True)
+ opener = urllib2.build_opener(urllib2.HTTPCookieProcessor(cj))
+ opener.addheaders = [ ("Current-Type", "application/x-www-form-urlencoded"),
+ ("Pragma", "no-cache"),
+ ("Content-Type", "application/x-www-form-urlencoded"),
+ ("Accept-Encoding", "deflate"),
+ ("Connection", "keep-alive"),
+ ("User-Agent", UA_MMOD)]
+
+ xbmc.log(str(base64.b64decode(media_token)))
+ payload = urllib.urlencode({'accessToken' : str(base64.b64decode(media_token)),
+ 'accessTokenType' : 'Adobe',
+ #'sessionId': '05a564d7c8622b5dd1c67f0ae737c5bb1515d66a~auth~win8~mml~100~1489621236122',
+ #'throttled' : 'no',
+ 'appData' : '{"clientTime":0}',
+ 'mvpd' : mvpd,
+ 'path' : FIND(stream_url,BASE_PATH,'master.m3u8')+'*'
+ })
+
+ resp = opener.open(url, payload)
+ response = resp.read()
+ resp.close()
+ hdnts = FIND(response,'<token>','</token>')
+ stream_url += '?hdnts='+hdnts
+ xbmc.log(stream_url)
+ stream_url = stream_url + '|User-Agent='+UA_MMOD
+
+
+ return stream_url
+
+
+def fetchStream(game_id,archive=None):
+ now = datetime.now()
+ url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/video/'+game_id+'_bk.json'
+ if archive == 'archive': url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/game/game_'+game_id+'.json'
+ now = datetime.now()
+ req = urllib2.Request(url)
+ req.add_header('Accept', '*/*')
+ req.add_header('User-Agent', UA_MMOD)
+ req.add_header('Accept-Language', 'en-us')
+ req.add_header('Accept-Encoding', 'deflate')
+
+ response = urllib2.urlopen(req)
+ json_source = json.load(response)
+ response.close()
+
+ if archive == 'archive':
+ stream_url = json_source['game']['videos']['video'][0]['connected']
+ else:
+ stream_url = json_source['connected1']
+
+
+ return stream_url
+
+
+def getAuthCookie():
+ mediaAuth = ''
+ try:
+ cj = cookielib.LWPCookieJar(os.path.join(ADDON_PATH_PROFILE, 'cookies.lwp'))
+ cj.load(os.path.join(ADDON_PATH_PROFILE, 'cookies.lwp'),ignore_discard=True)
+
+ #If authorization cookie is missing or stale, perform login
+ for cookie in cj:
+ if cookie.name == "mediaAuth" and not cookie.is_expired():
+ mediaAuth = 'mediaAuth='+cookie.value
+ except:
+ pass
+
+ return mediaAuth
+
+
+def addStream(name,link_url,title,game_id,icon=None,fanart=None):
+ ok=True
+ u=sys.argv[0]+"?url="+urllib.quote_plus(link_url)+"&mode="+str(104)+"&name="+urllib.quote_plus(name)+"&game_id="+urllib.quote_plus(str(game_id))
+
+ liz=xbmcgui.ListItem(name)
+ liz.setArt({'icon': ICON, 'thumb': ICON})
+
+ if fanart != None:
+ liz.setArt({'fanart': fanart})
+ else:
+ liz.setArt({'fanart': FANART})
+
+ liz.setProperty("IsPlayable", "true")
+ liz.setInfo( type="Video", infoLabels={ "Title": title } )
+
+ ok=xbmcplugin.addDirectoryItem(handle=int(sys.argv[1]),url=u,listitem=liz,isFolder=False)
+ xbmcplugin.setContent(addon_handle, 'episodes')
+ return ok
+
+def addDir(name,url,mode,icon,fanart=None):
+ ok=True
+ u=sys.argv[0]+"?url="+urllib.quote_plus(url)+"&mode="+str(mode)+"&name="+urllib.quote_plus(name)
+
+ liz=xbmcgui.ListItem(name)
+ liz.setArt({'icon': icon, 'thumb': icon})
+ liz.setInfo( type="Video", infoLabels={ "Title": name } )
+
+ if fanart != None:
+ liz.setArt({'fanart': fanart})
+ else:
+ liz.setArt({'fanart': FANART})
+
+ ok=xbmcplugin.addDirectoryItem(handle=int(sys.argv[1]),url=u,listitem=liz,isFolder=True)
+ xbmcplugin.setContent(int(sys.argv[1]), 'episodes')
+ return ok
+
+
+def addLink(name,url,icon,fanart=None):
+ ok=True
+
+ liz=xbmcgui.ListItem(name)
+ liz.setArt({'icon': icon, 'thumb': icon})
+ liz.setInfo( type="Video", infoLabels={ "Title": name } )
+ liz.setProperty("IsPlayable", "true")
+
+ if fanart != None:
+ liz.setArt({'fanart': fanart})
+ else:
+ liz.setArt({'fanart': FANART})
+
+
+ ok=xbmcplugin.addDirectoryItem(handle=int(sys.argv[1]),url=url,listitem=liz)
+ xbmcplugin.setContent(int(sys.argv[1]), 'episodes')
+ return ok
+
+
+# KODI ADDON GLOBALS
+ADDON_HANDLE = int(sys.argv[1])
+ADDON = xbmcaddon.Addon()
+ROOTDIR = ADDON.getAddonInfo('path')
+ADDON_ID = ADDON.getAddonInfo('id')
+ADDON_VERSION = ADDON.getAddonInfo('version')
+ADDON_PATH = xbmc.translatePath(ADDON.getAddonInfo('path'))
+ADDON_PATH_PROFILE = xbmc.translatePath(ADDON.getAddonInfo('profile'))
+KODI_VERSION = float(re.findall(r'\d{2}\.\d{1}', xbmc.getInfoLabel("System.BuildVersion"))[0])
+LOCAL_STRING = ADDON.getLocalizedString
+FANART = ROOTDIR+"/fanart.jpg"
+ICON = ROOTDIR+"/icon.png"
+
+#Main settings
+NO_SPOILERS = str(ADDON.getSetting(id="no_spoilers"))
+
+#User Agents
+UA_IPHONE = 'Mozilla/5.0 (iPhone; CPU iPhone OS 8_4 like Mac OS X) AppleWebKit/600.1.4 (KHTML, like Gecko) Mobile/12H143'
+UA_PC = 'Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/43.0.2357.81 Safari/537.36'
+UA_ADOBE_PASS = 'AdobePassNativeClient/1.9.4 (iPhone; U; CPU iPhone OS 9.2.1 like Mac OS X; en-us)'
+UA_MMOD = 'MML iOS/73 CFNetwork/808.3 Darwin/16.3.0'
+
+
+
+#Create Random Device ID and save it to a file
+fname = os.path.join(ADDON_PATH_PROFILE, 'device.id')
+if not os.path.isfile(fname):
+ if not os.path.exists(ADDON_PATH_PROFILE):
+ os.makedirs(ADDON_PATH_PROFILE)
+ new_device_id = ''.join([random.choice('0123456789abcdef') for x in range(64)])
+ device_file = open(fname,'w')
+ device_file.write(new_device_id)
+ device_file.close()
+
+fname = os.path.join(ADDON_PATH_PROFILE, 'device.id')
+device_file = open(fname,'r')
+DEVICE_ID = device_file.readline()
+device_file.close()
+
+
+#Event Colors
+FREE = 'FF43CD80'
+LIVE = 'FF00B7EB'
+UPCOMING = 'FFFFB266'
+FREE_UPCOMING = 'FFCC66FF' \ No newline at end of file
diff --git a/plugin.video.mmlive/resources/language/resource.language.en_gb/strings.po b/plugin.video.mmlive/resources/language/resource.language.en_gb/strings.po
new file mode 100644
index 0000000..6b0b0dc
--- /dev/null
+++ b/plugin.video.mmlive/resources/language/resource.language.en_gb/strings.po
@@ -0,0 +1,41 @@
+# Kodi Media Center language file
+# Addon Name: March Madness Live
+# Addon id: plugin.video.mmlive
+# Addon Provider: eracknaphobia
+msgid ""
+msgstr ""
+"Project-Id-Version: Kodi Addons\n"
+"Report-Msgid-Bugs-To: alanwww1@xbmc.org\n"
+"POT-Creation-Date: YEAR-MO-DA HO:MI+ZONE\n"
+"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
+"Last-Translator: Kodi Translation Team\n"
+"Language-Team: English (http://www.transifex.com/projects/p/xbmc-addons/language/en/)\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Language: en\n"
+"Plural-Forms: nplurals=2; plural=(n != 1);\n"
+
+msgctxt "#30000"
+msgid "Visual"
+msgstr ""
+
+msgctxt "#30100"
+msgid "Hide Scores"
+msgstr ""
+
+msgctxt "#30101"
+msgid "Off"
+msgstr ""
+
+msgctxt "#30102"
+msgid "On"
+msgstr ""
+
+msgctxt "#30103"
+msgid "Only Today's Games"
+msgstr ""
+
+msgctxt "#30104"
+msgid "Only Archive Games"
+msgstr "" \ No newline at end of file
diff --git a/plugin.video.mmlive/resources/settings.xml b/plugin.video.mmlive/resources/settings.xml
new file mode 100644
index 0000000..b173deb
--- /dev/null
+++ b/plugin.video.mmlive/resources/settings.xml
@@ -0,0 +1,7 @@
+<?xml version="1.0" encoding="utf-8" standalone="yes"?>
+<settings>
+ <!-- Visual -->
+ <category label="30000">
+ <setting id="no_spoilers" type="enum" label="30100" lvalues="30101|30102|30103|30104" default="30101" />
+ </category>
+</settings>