summaryrefslogtreecommitdiff
path: root/plugin.video.mmlive
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
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')
-rw-r--r--plugin.video.mmlive/LICENSE.txt339
-rw-r--r--plugin.video.mmlive/README.md5
-rw-r--r--plugin.video.mmlive/addon.xml24
-rw-r--r--plugin.video.mmlive/changelog.txt26
-rw-r--r--plugin.video.mmlive/fanart.jpgbin0 -> 682497 bytes
-rw-r--r--plugin.video.mmlive/icon.pngbin0 -> 16145 bytes
-rw-r--r--plugin.video.mmlive/main.py222
-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
12 files changed, 1326 insertions, 0 deletions
diff --git a/plugin.video.mmlive/LICENSE.txt b/plugin.video.mmlive/LICENSE.txt
new file mode 100644
index 0000000..d159169
--- /dev/null
+++ b/plugin.video.mmlive/LICENSE.txt
@@ -0,0 +1,339 @@
+ GNU GENERAL PUBLIC LICENSE
+ Version 2, June 1991
+
+ Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+ Preamble
+
+ The licenses for most software are designed to take away your
+freedom to share and change it. By contrast, the GNU General Public
+License is intended to guarantee your freedom to share and change free
+software--to make sure the software is free for all its users. This
+General Public License applies to most of the Free Software
+Foundation's software and to any other program whose authors commit to
+using it. (Some other Free Software Foundation software is covered by
+the GNU Lesser General Public License instead.) You can apply it to
+your programs, too.
+
+ When we speak of free software, we are referring to freedom, not
+price. Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+this service if you wish), that you receive source code or can get it
+if you want it, that you can change the software or use pieces of it
+in new free programs; and that you know you can do these things.
+
+ To protect your rights, we need to make restrictions that forbid
+anyone to deny you these rights or to ask you to surrender the rights.
+These restrictions translate to certain responsibilities for you if you
+distribute copies of the software, or if you modify it.
+
+ For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must give the recipients all the rights that
+you have. You must make sure that they, too, receive or can get the
+source code. And you must show them these terms so they know their
+rights.
+
+ We protect your rights with two steps: (1) copyright the software, and
+(2) offer you this license which gives you legal permission to copy,
+distribute and/or modify the software.
+
+ Also, for each author's protection and ours, we want to make certain
+that everyone understands that there is no warranty for this free
+software. If the software is modified by someone else and passed on, we
+want its recipients to know that what they have is not the original, so
+that any problems introduced by others will not reflect on the original
+authors' reputations.
+
+ Finally, any free program is threatened constantly by software
+patents. We wish to avoid the danger that redistributors of a free
+program will individually obtain patent licenses, in effect making the
+program proprietary. To prevent this, we have made it clear that any
+patent must be licensed for everyone's free use or not licensed at all.
+
+ The precise terms and conditions for copying, distribution and
+modification follow.
+
+ GNU GENERAL PUBLIC LICENSE
+ TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
+
+ 0. This License applies to any program or other work which contains
+a notice placed by the copyright holder saying it may be distributed
+under the terms of this General Public License. The "Program", below,
+refers to any such program or work, and a "work based on the Program"
+means either the Program or any derivative work under copyright law:
+that is to say, a work containing the Program or a portion of it,
+either verbatim or with modifications and/or translated into another
+language. (Hereinafter, translation is included without limitation in
+the term "modification".) Each licensee is addressed as "you".
+
+Activities other than copying, distribution and modification are not
+covered by this License; they are outside its scope. The act of
+running the Program is not restricted, and the output from the Program
+is covered only if its contents constitute a work based on the
+Program (independent of having been made by running the Program).
+Whether that is true depends on what the Program does.
+
+ 1. You may copy and distribute verbatim copies of the Program's
+source code as you receive it, in any medium, provided that you
+conspicuously and appropriately publish on each copy an appropriate
+copyright notice and disclaimer of warranty; keep intact all the
+notices that refer to this License and to the absence of any warranty;
+and give any other recipients of the Program a copy of this License
+along with the Program.
+
+You may charge a fee for the physical act of transferring a copy, and
+you may at your option offer warranty protection in exchange for a fee.
+
+ 2. You may modify your copy or copies of the Program or any portion
+of it, thus forming a work based on the Program, and copy and
+distribute such modifications or work under the terms of Section 1
+above, provided that you also meet all of these conditions:
+
+ a) You must cause the modified files to carry prominent notices
+ stating that you changed the files and the date of any change.
+
+ b) You must cause any work that you distribute or publish, that in
+ whole or in part contains or is derived from the Program or any
+ part thereof, to be licensed as a whole at no charge to all third
+ parties under the terms of this License.
+
+ c) If the modified program normally reads commands interactively
+ when run, you must cause it, when started running for such
+ interactive use in the most ordinary way, to print or display an
+ announcement including an appropriate copyright notice and a
+ notice that there is no warranty (or else, saying that you provide
+ a warranty) and that users may redistribute the program under
+ these conditions, and telling the user how to view a copy of this
+ License. (Exception: if the Program itself is interactive but
+ does not normally print such an announcement, your work based on
+ the Program is not required to print an announcement.)
+
+These requirements apply to the modified work as a whole. If
+identifiable sections of that work are not derived from the Program,
+and can be reasonably considered independent and separate works in
+themselves, then this License, and its terms, do not apply to those
+sections when you distribute them as separate works. But when you
+distribute the same sections as part of a whole which is a work based
+on the Program, the distribution of the whole must be on the terms of
+this License, whose permissions for other licensees extend to the
+entire whole, and thus to each and every part regardless of who wrote it.
+
+Thus, it is not the intent of this section to claim rights or contest
+your rights to work written entirely by you; rather, the intent is to
+exercise the right to control the distribution of derivative or
+collective works based on the Program.
+
+In addition, mere aggregation of another work not based on the Program
+with the Program (or with a work based on the Program) on a volume of
+a storage or distribution medium does not bring the other work under
+the scope of this License.
+
+ 3. You may copy and distribute the Program (or a work based on it,
+under Section 2) in object code or executable form under the terms of
+Sections 1 and 2 above provided that you also do one of the following:
+
+ a) Accompany it with the complete corresponding machine-readable
+ source code, which must be distributed under the terms of Sections
+ 1 and 2 above on a medium customarily used for software interchange; or,
+
+ b) Accompany it with a written offer, valid for at least three
+ years, to give any third party, for a charge no more than your
+ cost of physically performing source distribution, a complete
+ machine-readable copy of the corresponding source code, to be
+ distributed under the terms of Sections 1 and 2 above on a medium
+ customarily used for software interchange; or,
+
+ c) Accompany it with the information you received as to the offer
+ to distribute corresponding source code. (This alternative is
+ allowed only for noncommercial distribution and only if you
+ received the program in object code or executable form with such
+ an offer, in accord with Subsection b above.)
+
+The source code for a work means the preferred form of the work for
+making modifications to it. For an executable work, complete source
+code means all the source code for all modules it contains, plus any
+associated interface definition files, plus the scripts used to
+control compilation and installation of the executable. However, as a
+special exception, the source code distributed need not include
+anything that is normally distributed (in either source or binary
+form) with the major components (compiler, kernel, and so on) of the
+operating system on which the executable runs, unless that component
+itself accompanies the executable.
+
+If distribution of executable or object code is made by offering
+access to copy from a designated place, then offering equivalent
+access to copy the source code from the same place counts as
+distribution of the source code, even though third parties are not
+compelled to copy the source along with the object code.
+
+ 4. You may not copy, modify, sublicense, or distribute the Program
+except as expressly provided under this License. Any attempt
+otherwise to copy, modify, sublicense or distribute the Program is
+void, and will automatically terminate your rights under this License.
+However, parties who have received copies, or rights, from you under
+this License will not have their licenses terminated so long as such
+parties remain in full compliance.
+
+ 5. You are not required to accept this License, since you have not
+signed it. However, nothing else grants you permission to modify or
+distribute the Program or its derivative works. These actions are
+prohibited by law if you do not accept this License. Therefore, by
+modifying or distributing the Program (or any work based on the
+Program), you indicate your acceptance of this License to do so, and
+all its terms and conditions for copying, distributing or modifying
+the Program or works based on it.
+
+ 6. Each time you redistribute the Program (or any work based on the
+Program), the recipient automatically receives a license from the
+original licensor to copy, distribute or modify the Program subject to
+these terms and conditions. You may not impose any further
+restrictions on the recipients' exercise of the rights granted herein.
+You are not responsible for enforcing compliance by third parties to
+this License.
+
+ 7. If, as a consequence of a court judgment or allegation of patent
+infringement or for any other reason (not limited to patent issues),
+conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License. If you cannot
+distribute so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you
+may not distribute the Program at all. For example, if a patent
+license would not permit royalty-free redistribution of the Program by
+all those who receive copies directly or indirectly through you, then
+the only way you could satisfy both it and this License would be to
+refrain entirely from distribution of the Program.
+
+If any portion of this section is held invalid or unenforceable under
+any particular circumstance, the balance of the section is intended to
+apply and the section as a whole is intended to apply in other
+circumstances.
+
+It is not the purpose of this section to induce you to infringe any
+patents or other property right claims or to contest validity of any
+such claims; this section has the sole purpose of protecting the
+integrity of the free software distribution system, which is
+implemented by public license practices. Many people have made
+generous contributions to the wide range of software distributed
+through that system in reliance on consistent application of that
+system; it is up to the author/donor to decide if he or she is willing
+to distribute software through any other system and a licensee cannot
+impose that choice.
+
+This section is intended to make thoroughly clear what is believed to
+be a consequence of the rest of this License.
+
+ 8. If the distribution and/or use of the Program is restricted in
+certain countries either by patents or by copyrighted interfaces, the
+original copyright holder who places the Program under this License
+may add an explicit geographical distribution limitation excluding
+those countries, so that distribution is permitted only in or among
+countries not thus excluded. In such case, this License incorporates
+the limitation as if written in the body of this License.
+
+ 9. The Free Software Foundation may publish revised and/or new versions
+of the General Public License from time to time. Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+Each version is given a distinguishing version number. If the Program
+specifies a version number of this License which applies to it and "any
+later version", you have the option of following the terms and conditions
+either of that version or of any later version published by the Free
+Software Foundation. If the Program does not specify a version number of
+this License, you may choose any version ever published by the Free Software
+Foundation.
+
+ 10. If you wish to incorporate parts of the Program into other free
+programs whose distribution conditions are different, write to the author
+to ask for permission. For software which is copyrighted by the Free
+Software Foundation, write to the Free Software Foundation; we sometimes
+make exceptions for this. Our decision will be guided by the two goals
+of preserving the free status of all derivatives of our free software and
+of promoting the sharing and reuse of software generally.
+
+ NO WARRANTY
+
+ 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
+FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
+OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
+PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
+OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
+TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
+PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
+REPAIR OR CORRECTION.
+
+ 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
+REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
+INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
+OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
+TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
+YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
+PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGES.
+
+ END OF TERMS AND CONDITIONS
+
+ How to Apply These Terms to Your New Programs
+
+ If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+ To do so, attach the following notices to the program. It is safest
+to attach them to the start of each source file to most effectively
+convey the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+ <one line to give the program's name and a brief idea of what it does.>
+ Copyright (C) <year> <name of author>
+
+ This program 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 2 of the License, or
+ (at your option) any later version.
+
+ This program 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 this program; if not, write to the Free Software Foundation, Inc.,
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+
+Also add information on how to contact you by electronic and paper mail.
+
+If the program is interactive, make it output a short notice like this
+when it starts in an interactive mode:
+
+ Gnomovision version 69, Copyright (C) year name of author
+ Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+ This is free software, and you are welcome to redistribute it
+ under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License. Of course, the commands you use may
+be called something other than `show w' and `show c'; they could even be
+mouse-clicks or menu items--whatever suits your program.
+
+You should also get your employer (if you work as a programmer) or your
+school, if any, to sign a "copyright disclaimer" for the program, if
+necessary. Here is a sample; alter the names:
+
+ Yoyodyne, Inc., hereby disclaims all copyright interest in the program
+ `Gnomovision' (which makes passes at compilers) written by James Hacker.
+
+ <signature of Ty Coon>, 1 April 1989
+ Ty Coon, President of Vice
+
+This General Public License does not permit incorporating your program into
+proprietary programs. If your program is a subroutine library, you may
+consider it more useful to permit linking proprietary applications with the
+library. If this is what you want to do, use the GNU Lesser General
+Public License instead of this License.
diff --git a/plugin.video.mmlive/README.md b/plugin.video.mmlive/README.md
new file mode 100644
index 0000000..78ac825
--- /dev/null
+++ b/plugin.video.mmlive/README.md
@@ -0,0 +1,5 @@
+plugin.video.mmlive
+======================
+
+KODI plugin March Madness Live
+
diff --git a/plugin.video.mmlive/addon.xml b/plugin.video.mmlive/addon.xml
new file mode 100644
index 0000000..f82a157
--- /dev/null
+++ b/plugin.video.mmlive/addon.xml
@@ -0,0 +1,24 @@
+<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
+<addon id="plugin.video.mmlive" name="March Madness Live" version="2017.3.19" provider-name="eracknaphobia">
+ <requires>
+ <import addon="xbmc.python" version="2.25.0"/>
+ </requires>
+ <extension point="xbmc.python.pluginsource" library="main.py">
+ <provides>video</provides>
+ </extension>
+
+ <extension point="xbmc.addon.metadata">
+ <platform>all</platform>
+ <summary lang="en_GB"></summary>
+ <description lang="en_GB">The National Collegiate Athletic Association (NCAA) Men's Division I Basketball Tournament is a single-elimination tournament played each spring in the United States, currently featuring 68 college basketball teams, to determine the national championship of the major college basketball teams.</description>
+ <disclaimer lang="en_GB"></disclaimer>
+ <language>en</language>
+ <platform>all</platform>
+ <license>GNU GENERAL PUBLIC LICENSE. Version 2, June 1991</license>
+ <forum>http://forum.kodi.tv/showthread.php?tid=264888</forum>
+ <website></website>
+ <email></email>
+ <source>https://github.com/eracknaphobia/plugin.video.mmlive/tree/krypton</source>
+ </extension>
+</addon>
+
diff --git a/plugin.video.mmlive/changelog.txt b/plugin.video.mmlive/changelog.txt
new file mode 100644
index 0000000..46fd9a9
--- /dev/null
+++ b/plugin.video.mmlive/changelog.txt
@@ -0,0 +1,26 @@
+2017.3.19
+- Bumped python version for Krypton
+
+2017.3.18
+- Fixed completed games not showing final
+- Fixed some stream parameter issues
+- Code clean up
+
+2017.3.17
+- Added a deauthorize option
+- Added seed number to team name
+- Archive games are ~2min Recaps (for now)
+- Removed httplib2 requirement
+
+2017.3.16
+- Added Classic Games Section
+- Switched to registration code login
+
+2016.3.18.3
+- Fix Cox cable login
+
+2016.3.18.2
+- Fix settings bug
+
+2016.3.18
+- Beta released \ No newline at end of file
diff --git a/plugin.video.mmlive/fanart.jpg b/plugin.video.mmlive/fanart.jpg
new file mode 100644
index 0000000..f5e3512
--- /dev/null
+++ b/plugin.video.mmlive/fanart.jpg
Binary files differ
diff --git a/plugin.video.mmlive/icon.png b/plugin.video.mmlive/icon.png
new file mode 100644
index 0000000..330d9a3
--- /dev/null
+++ b/plugin.video.mmlive/icon.png
Binary files differ
diff --git a/plugin.video.mmlive/main.py b/plugin.video.mmlive/main.py
new file mode 100644
index 0000000..7cbbdc8
--- /dev/null
+++ b/plugin.video.mmlive/main.py
@@ -0,0 +1,222 @@
+from resources.globals import *
+from resources.adobepass import ADOBE
+
+#Add-on specific Adobepass variables
+SERVICE_VARS = {'requestor_id':'MML',
+ 'public_key':'XfId78vskMBegCUx9fuiNQL3XvxP3SzN',
+ 'private_key':'60OKORsYmOkUMgDm',
+ 'activate_url':'ncaa.com/activate'
+ }
+
+def categories():
+ addDir('Today\'s Games','/live',1,ICON,FANART)
+ addDir('Archive Games','/live',2,ICON,FANART)
+ addDir('Classic Games','/classic',3,ICON,FANART)
+ addDir('Deauthorize this Device','/deauth',4,ICON,FANART)
+
+
+def todaysGames(archive=None):
+ now = datetime.now()
+ url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/bracket.json'
+
+ req = urllib2.Request(url)
+ 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()
+
+ tourn_day = json_source['bracket']['tournDay']
+ teams = getTournamentInfo()
+
+
+ if not archive:
+ setTodaysStream(tourn_day, json_source, teams)
+ else:
+ setArchiveStreams(tourn_day, json_source, teams)
+
+
+def setTodaysStream(tourn_day, json_source, teams):
+ tomorrow = str(int(tourn_day) + 86400)
+ try:
+ current_games = getCurrentInfo()
+ except:
+ pass
+
+ #Sort By Start Time
+ json_source = sorted(json_source['bracket']['game'],key=lambda x:x['time'])
+
+ for game in json_source:
+ if game['time'] >= tourn_day and game['time'] < tomorrow:
+ if game['tmH'] != '' and game['tmV'] != '':
+ game_id = game['id']
+ hTeam = getTeamInfo(teams, game['tmH'])
+ vTeam = getTeamInfo(teams,game['tmV'])
+ game_time = time.strftime('%I:%M %p', time.localtime(int(game['time']))).lstrip('0')
+ state = game['state']
+ archive_video = game['rcpV']
+
+ title = vTeam['school'] + ' vs ' + hTeam['school']
+
+ if NO_SPOILERS == '1' or NO_SPOILERS == '2':
+ name = title
+ else:
+ name = '#'+ vTeam['seed']+ ' ' + vTeam['school'] + ' ' + colorString(game['ptsV'], SCORE_COLOR) + ' vs #'+ hTeam['seed']+ ' ' + hTeam['school'] + ' ' + colorString(game['ptsH'], SCORE_COLOR)
+
+ if state == "1":
+ name = colorString(game_time, UPCOMING) + ' ' + name
+ elif state == "4":
+ name = colorString("FINAL", FINAL) + ' ' + name
+ else:
+ clock = getGameClock(current_games, game_id)
+ if clock == '':
+ clock = 'LIVE'
+ name = colorString(clock,GAMETIME_COLOR) + ' ' + name
+
+
+ link_url = ''
+ addStream(name,link_url,title,game_id)
+
+
+def classicGames():
+ now = datetime.now()
+ url = 'http://data.ncaa.com/mml/'+str(now.year)+'/mobile/vod/classic_games.json'
+ req = urllib2.Request(url)
+ 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()
+
+ for game in json_source['videos']:
+ title = game['title']
+ url = game['connected'] + '|User-Agent='+UA_MMOD
+ icon = game['thumbnails']['large']
+ fanart = game['thumbnails']['raw']
+ addLink(title,url,icon,fanart)
+
+
+
+def setArchiveStreams(tourn_day, json_source, teams):
+ json_source = sorted(json_source['bracket']['game'],key=lambda x:x['time'], reverse=True)
+ for game in json_source:
+ if game['time'] < tourn_day:
+ if game['tmH'] != '' and game['tmV'] != '':
+ game_id = game['id']
+ hTeam = getTeamInfo(teams, game['tmH'])
+ vTeam = getTeamInfo(teams,game['tmV'])
+ game_time = time.strftime('%I:%M %p', time.localtime(int(game['time']))).lstrip('0')
+ live_video = game['video']
+ archive_video = game['rcpV']
+
+ title = vTeam['school'] + ' vs ' + hTeam['school']
+
+ if NO_SPOILERS == '1' or NO_SPOILERS == '3':
+ name = title
+ else:
+ name = '#'+ vTeam['seed']+ ' ' + vTeam['school'] + ' ' + colorString(game['ptsV'], SCORE_COLOR) + ' vs ' + '#'+ hTeam['seed']+ ' ' + hTeam['school'] + ' ' + colorString(game['ptsH'], SCORE_COLOR)
+
+ name = colorString('FINAL ',FINAL) + name
+ link_url = 'archive'
+ addStream(name,link_url,title,game_id)
+
+
+def startStream(game_id):
+ stream_url = fetchStream(game_id,addon_url)
+
+ if addon_url == 'archive':
+ playable_stream = stream_url + '|User-Agent='+UA_MMOD
+ else:
+ adobe = ADOBE(SERVICE_VARS)
+ resource_id = 'truTV'
+ mvpd = adobe.authorizeDevice(resource_id)
+ media_token = adobe.mediaToken(resource_id)
+ playable_stream = tokenTurner(media_token,stream_url,mvpd)
+
+ listitem = xbmcgui.ListItem(path=playable_stream)
+ xbmcplugin.setResolvedUrl(handle=addon_handle, succeeded=True, listitem=listitem)
+
+
+
+def get_params():
+ param=[]
+ paramstring=sys.argv[2]
+ if len(paramstring)>=2:
+ params=sys.argv[2]
+ cleanedparams=params.replace('?','')
+ if (params[len(params)-1]=='/'):
+ params=params[0:len(params)-2]
+ pairsofparams=cleanedparams.split('&')
+ param={}
+ for i in range(len(pairsofparams)):
+ splitparams={}
+ splitparams=pairsofparams[i].split('=')
+ if (len(splitparams))==2:
+ param[splitparams[0]]=splitparams[1]
+
+ return param
+
+
+params=get_params()
+addon_url=None
+name=None
+mode=None
+game_id=None
+icon_image = None
+
+try:
+ addon_url=urllib.unquote_plus(params["url"])
+except:
+ pass
+try:
+ name=urllib.unquote_plus(params["name"])
+except:
+ pass
+try:
+ mode=int(params["mode"])
+except:
+ pass
+try:
+ game_id=urllib.unquote_plus(params["game_id"])
+except:
+ pass
+try:
+ icon_image=urllib.unquote_plus(params["icon_image"])
+except:
+ pass
+
+
+if mode==None:
+ categories()
+elif mode==1:
+ todaysGames()
+elif mode==2:
+ todaysGames(archive=True)
+elif mode==3:
+ classicGames()
+elif mode==4:
+ msg = 'Are you sure you wish to deauthorize this device?'
+ dialog = xbmcgui.Dialog()
+ answer = dialog.yesno('Deauthorize Devices', msg)
+ if answer:
+ adobe = ADOBE(SERVICE_VARS)
+ adobe.deauthorizeDevice()
+ sys.exit()
+
+elif mode==104:
+ startStream(game_id)
+
+
+#Don't cache todays games
+if mode==1:
+ xbmcplugin.endOfDirectory(addon_handle, cacheToDisc=False)
+else:
+ xbmcplugin.endOfDirectory(addon_handle)
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>