Merge pull request #30 from warwickh/master
Merge Matrix compatible to master
This commit is contained in:
		| @ -1,5 +1,14 @@ | |||||||
| # Changelog | # Changelog | ||||||
|  |  | ||||||
|  | ## v3.0.0 | ||||||
|  | Released 29th June 2021 (by warwickh) | ||||||
|  | * Basic update to provide Matrix compatility. Not tested on Kodi below v19 | ||||||
|  | * Updates simpleplugin to latest version (3.0.0) https://github.com/vlmaksime/script.module.simpleplugin | ||||||
|  | * Moves some legacy simpleplugin static routines into main.py | ||||||
|  | * Removes dependancy on libsonic_extra by moving some walk functions into main.py | ||||||
|  | * Updates libsonic to latest version and adds functions for returning raw url for populating menus | ||||||
|  | * Move to version 3+ for diffferentiation from Leia compatible version | ||||||
|  |  | ||||||
| ## v2.0.8 | ## v2.0.8 | ||||||
| Released 29th November 2017 (by Heruwar) | Released 29th November 2017 (by Heruwar) | ||||||
| * Fixes a security issue where the password is sent as plaintext in the URL query parameters when methods from libsonic_extas are used.  | * Fixes a security issue where the password is sent as plaintext in the URL query parameters when methods from libsonic_extas are used.  | ||||||
|  | |||||||
							
								
								
									
										22
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										22
									
								
								README.md
									
									
									
									
									
								
							| @ -1,9 +1,15 @@ | |||||||
| # Subsonic | # Subsonic | ||||||
| Kodi plugin to stream, star and download music from Subsonic. | Kodi plugin to stream, star and download music from Subsonic. | ||||||
|  |  | ||||||
| For feature requests / issues: | For feature requests / issues: | ||||||
| https://github.com/gordielachance/plugin.audio.subsonic/issues | https://github.com/warwickh/plugin.audio.subsonic/issues | ||||||
|  |  | ||||||
| Contributions are welcome: | Contributions are welcome: | ||||||
| https://github.com/gordielachance/plugin.audio.subsonic | https://github.com/warwickh/plugin.audio.subsonic | ||||||
|  |  | ||||||
|  | Master branch updated to support Kodi 19 Matrix | ||||||
|  |  | ||||||
|  | Leia compatible version available in alternate branch | ||||||
|  |  | ||||||
| ## Features | ## Features | ||||||
| * Browse by artist, albums (newest/most played/recently played/random), tracks (starred/random), and playlists | * Browse by artist, albums (newest/most played/recently played/random), tracks (starred/random), and playlists | ||||||
| @ -11,10 +17,17 @@ https://github.com/gordielachance/plugin.audio.subsonic | |||||||
| * Star songs | * Star songs | ||||||
|  |  | ||||||
| ## Installation | ## Installation | ||||||
|  | * Click the code button and download | ||||||
|  | * Enable unknown sources and install from zip in Kodi | ||||||
|  |  | ||||||
|  | or | ||||||
|  |  | ||||||
| * Navigate to your `.kodi/addons/` folder | * Navigate to your `.kodi/addons/` folder | ||||||
| * Clone this repository: `git clone https://github.com/gordielachance/plugin.audio.subsonic.git` | * Clone this repository: `git clone https://github.com/warwickh/plugin.audio.subsonic.git` | ||||||
| * (Re)start Kodi. | * (Re)start Kodi. | ||||||
|  |  | ||||||
|  | Note: You may need to install dependencies manually if installing this way | ||||||
|  |  | ||||||
| ## TODO | ## TODO | ||||||
| * Scrobble to Last.FM (http://forum.kodi.tv/showthread.php?tid=195597&pid=2429362#pid2429362) | * Scrobble to Last.FM (http://forum.kodi.tv/showthread.php?tid=195597&pid=2429362#pid2429362) | ||||||
| * Improve the caching system | * Improve the caching system | ||||||
| @ -24,7 +37,8 @@ https://github.com/gordielachance/plugin.audio.subsonic | |||||||
| See the `LICENSE` file. | See the `LICENSE` file. | ||||||
|  |  | ||||||
| Additional copyright notices: | Additional copyright notices: | ||||||
|  | * [Previous version of this plugin](https://github.com/gordielachance/plugin.audio.subsonic) by gordielachance | ||||||
| * [Previous version of this plugin](https://github.com/basilfx/plugin.audio.subsonic) by basilfx | * [Previous version of this plugin](https://github.com/basilfx/plugin.audio.subsonic) by basilfx | ||||||
| * [SimplePlugin](https://github.com/romanvm/script.module.simpleplugin/stargazers) by romanvm | * [SimplePlugin](https://github.com/romanvm/script.module.simpleplugin/stargazers) by romanvm now at [SimplePlugin3](https://github.com/vlmaksime/script.module.simpleplugin) | ||||||
| * The original [SubKodi](https://github.com/DarkAllMan/SubKodi) plugin | * The original [SubKodi](https://github.com/DarkAllMan/SubKodi) plugin | ||||||
| * [`py-sonic`](https://github.com/crustymonkey/py-sonic) Python module | * [`py-sonic`](https://github.com/crustymonkey/py-sonic) Python module | ||||||
|  | |||||||
							
								
								
									
										23
									
								
								addon.xml
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								addon.xml
									
									
									
									
									
								
							| @ -1,8 +1,9 @@ | |||||||
| <?xml version="1.0" encoding="UTF-8" standalone="yes"?> | <?xml version="1.0" encoding="UTF-8" standalone="yes"?> | ||||||
| <addon id="plugin.audio.subsonic" name="Subsonic" version="2.0.8" provider-name="BasilFX,grosbouff,silascutler,Heruwar"> | <addon id="plugin.audio.subsonic" name="Subsonic" version="3.0.0" provider-name="BasilFX,grosbouff,silascutler,Heruwar,warwickh"> | ||||||
| <requires> | <requires> | ||||||
|     <import addon="xbmc.python" version="2.14.0"/> |     <import addon="xbmc.python" version="3.0.0"/> | ||||||
|     <import addon="script.module.dateutil" version="2.4.2"/> |     <import addon="script.module.dateutil" version="2.4.2"/> | ||||||
|  |     <import addon="script.module.future" version="0.18.2"/> | ||||||
| </requires> | </requires> | ||||||
| <extension point="xbmc.python.pluginsource" library="main.py"> | <extension point="xbmc.python.pluginsource" library="main.py"> | ||||||
|   <provides>audio</provides> |   <provides>audio</provides> | ||||||
| @ -14,31 +15,35 @@ | |||||||
|         <description lang="en"> |         <description lang="en"> | ||||||
|             Stream, star and download your tunes, directly to Kodi ! |             Stream, star and download your tunes, directly to Kodi ! | ||||||
|             For feature requests / issues: |             For feature requests / issues: | ||||||
|             https://github.com/gordielachance/plugin.audio.subsonic/issues |             https://github.com/warwickh/plugin.audio.subsonic/issues | ||||||
|             Contributions are welcome: |             Contributions are welcome: | ||||||
|             https://github.com/gordielachance/plugin.audio.subsonic |             https://github.com/warwickh/plugin.audio.subsonic | ||||||
|         </description> |         </description> | ||||||
|         <description lang="fr"> |         <description lang="fr"> | ||||||
|             Jouez, marquez vos favoris et téléchargez votre musique, directement dans Kodi ! |             Jouez, marquez vos favoris et téléchargez votre musique, directement dans Kodi ! | ||||||
|             Pour les demandes et problèmes : |             Pour les demandes et problèmes : | ||||||
|             https://github.com/gordielachance/plugin.audio.subsonic/issues |             https://github.com/warwickh/plugin.audio.subsonic/issues | ||||||
|             Les contributions sont les bienvenues : |             Les contributions sont les bienvenues : | ||||||
|             https://github.com/gordielachance/plugin.audio.subsonic |             https://github.com/warwickh/plugin.audio.subsonic | ||||||
|         </description> |         </description> | ||||||
| 		<description lang="de"> | 		<description lang="de"> | ||||||
|             Streame, bewerte und downloade deine Medien direkt in Kodi ! |             Streame, bewerte und downloade deine Medien direkt in Kodi ! | ||||||
| 	    Für neue Eigentschaften oder Fehler: | 	    Für neue Eigentschaften oder Fehler: | ||||||
|             https://github.com/gordielachance/plugin.audio.subsonic/issues |             https://github.com/warwickh/plugin.audio.subsonic/issues | ||||||
|             Beihilfe ist Willkommen: |             Beihilfe ist Willkommen: | ||||||
|             https://github.com/gordielachance/plugin.audio.subsonic |             https://github.com/warwickh/plugin.audio.subsonic | ||||||
|         </description> |         </description> | ||||||
|  |         <assets> | ||||||
|  |             <icon>icon.png</icon> | ||||||
|  |             <fanart>fanart.png</fanart> | ||||||
|  |         </assets> | ||||||
|         <disclaimer lang="en"></disclaimer> |         <disclaimer lang="en"></disclaimer> | ||||||
|         <language>multi</language> |         <language>multi</language> | ||||||
|         <platform>all</platform> |         <platform>all</platform> | ||||||
|         <license>MIT</license> |         <license>MIT</license> | ||||||
|         <forum></forum> |         <forum></forum> | ||||||
|         <website>http://www.subsonic.org</website> |         <website>http://www.subsonic.org</website> | ||||||
| 	<source>https://github.com/gordielachance/plugin.audio.subsonic</source> | 	<source>https://github.com/warwickh/plugin.audio.subsonic</source> | ||||||
|         <email></email> |         <email></email> | ||||||
|     </extension> |     </extension> | ||||||
| </addon> | </addon> | ||||||
|  | |||||||
							
								
								
									
										
											BIN
										
									
								
								fanart.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								fanart.jpg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 124 KiB | 
| @ -1,32 +1,24 @@ | |||||||
| """ | """ | ||||||
| This file is part of py-sonic. | This file is part of py-sonic. | ||||||
|  |  | ||||||
| py-sonic is free software: you can redistribute it and/or modify | py-sonic is free software: you can redistribute it and/or modify | ||||||
| it under the terms of the GNU General Public License as published by | it under the terms of the GNU General Public License as published by | ||||||
| the Free Software Foundation, either version 3 of the License, or | the Free Software Foundation, either version 3 of the License, or | ||||||
| (at your option) any later version. | (at your option) any later version. | ||||||
|  |  | ||||||
| py-sonic is distributed in the hope that it will be useful, | py-sonic is distributed in the hope that it will be useful, | ||||||
| but WITHOUT ANY WARRANTY; without even the implied warranty of | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
| GNU General Public License for more details. | GNU General Public License for more details. | ||||||
|  |  | ||||||
| You should have received a copy of the GNU General Public License | You should have received a copy of the GNU General Public License | ||||||
| along with py-sonic.  If not, see <http://www.gnu.org/licenses/> | along with py-sonic.  If not, see <http://www.gnu.org/licenses/> | ||||||
|  |  | ||||||
| For information on method calls, see 'pydoc libsonic.connection' | For information on method calls, see 'pydoc libsonic.connection' | ||||||
|  |  | ||||||
| ---------- | ---------- | ||||||
| Basic example: | Basic example: | ||||||
| ---------- | ---------- | ||||||
|  |  | ||||||
| import libsonic | import libsonic | ||||||
|  |  | ||||||
| conn = libsonic.Connection('http://localhost' , 'admin' , 'password') | conn = libsonic.Connection('http://localhost' , 'admin' , 'password') | ||||||
| print conn.ping() | print conn.ping() | ||||||
|  |  | ||||||
| """ | """ | ||||||
|  |  | ||||||
| from connection import * | from .connection import * | ||||||
|  |  | ||||||
| __version__ = '0.6.2' | __version__ = '0.7.9' | ||||||
|  | |||||||
| @ -15,23 +15,28 @@ You should have received a copy of the GNU General Public License | |||||||
| along with py-sonic.  If not, see <http://www.gnu.org/licenses/> | along with py-sonic.  If not, see <http://www.gnu.org/licenses/> | ||||||
| """ | """ | ||||||
|  |  | ||||||
| from urllib import urlencode | from libsonic.errors import * | ||||||
| from .errors import * |  | ||||||
| from pprint import pprint |  | ||||||
| from cStringIO import StringIO |  | ||||||
| from netrc import netrc | from netrc import netrc | ||||||
| from hashlib import md5 | from hashlib import md5 | ||||||
| import json, urllib2, httplib, logging, socket, ssl, sys, os | import urllib.request | ||||||
|  | import urllib.error | ||||||
|  | from http import client as http_client | ||||||
|  | from urllib.parse import urlencode | ||||||
|  | from io import StringIO | ||||||
|  |  | ||||||
| API_VERSION = '1.14.0' | import json | ||||||
|  | import logging | ||||||
|  | import socket | ||||||
|  | import ssl | ||||||
|  | import sys | ||||||
|  | import os | ||||||
|  | import xbmc | ||||||
|  |  | ||||||
|  | API_VERSION = '1.16.1' | ||||||
|  |  | ||||||
| logger = logging.getLogger(__name__) | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
| class HTTPSConnectionChain(httplib.HTTPSConnection): | class HTTPSConnectionChain(http_client.HTTPSConnection): | ||||||
|     _preferred_ssl_protos = sorted([ p for p in dir(ssl) |  | ||||||
|         if p.startswith('PROTOCOL_') ], reverse=True) |  | ||||||
|     _ssl_working_proto = None |  | ||||||
|  |  | ||||||
|     def _create_sock(self): |     def _create_sock(self): | ||||||
|         sock = socket.create_connection((self.host, self.port), self.timeout) |         sock = socket.create_connection((self.host, self.port), self.timeout) | ||||||
|         if self._tunnel_host: |         if self._tunnel_host: | ||||||
| @ -40,38 +45,21 @@ class HTTPSConnectionChain(httplib.HTTPSConnection): | |||||||
|         return sock |         return sock | ||||||
|  |  | ||||||
|     def connect(self): |     def connect(self): | ||||||
|         if self._ssl_working_proto is not None: |  | ||||||
|             # If we have a working proto, let's use that straight away |  | ||||||
|             logger.debug("Using known working proto: '%s'", |  | ||||||
|                          self._ssl_working_proto) |  | ||||||
|         sock = self._create_sock() |         sock = self._create_sock() | ||||||
|             self.sock = ssl.wrap_socket(sock, self.key_file, self.cert_file, |  | ||||||
|                 ssl_version=self._ssl_working_proto) |  | ||||||
|             return |  | ||||||
|  |  | ||||||
|         # Try connecting via the different SSL protos in preference order |  | ||||||
|         for proto_name in self._preferred_ssl_protos: |  | ||||||
|             sock = self._create_sock() |  | ||||||
|             proto = getattr(ssl, proto_name, None) |  | ||||||
|         try: |         try: | ||||||
|                 self.sock = ssl.wrap_socket(sock, self.key_file, self.cert_file, |             self.sock = ssl.create_default_context().wrap_socket(sock, | ||||||
|                     ssl_version=proto) |                 server_hostname=self.host) | ||||||
|         except: |         except: | ||||||
|             sock.close() |             sock.close() | ||||||
|             else: |  | ||||||
|                 # Cache the working ssl version |  | ||||||
|                 HTTPSConnectionChain._ssl_working_proto = proto |  | ||||||
|                 break |  | ||||||
|  |  | ||||||
|  | class HTTPSHandlerChain(urllib.request.HTTPSHandler): | ||||||
| class HTTPSHandlerChain(urllib2.HTTPSHandler): |  | ||||||
|     def https_open(self, req): |     def https_open(self, req): | ||||||
|         return self.do_open(HTTPSConnectionChain, req) |         return self.do_open(HTTPSConnectionChain, req) | ||||||
|  |  | ||||||
| # install opener | # install opener | ||||||
| urllib2.install_opener(urllib2.build_opener(HTTPSHandlerChain())) | urllib.request.install_opener(urllib.request.build_opener(HTTPSHandlerChain())) | ||||||
|  |  | ||||||
| class PysHTTPRedirectHandler(urllib2.HTTPRedirectHandler): | class PysHTTPRedirectHandler(urllib.request.HTTPRedirectHandler): | ||||||
|     """ |     """ | ||||||
|     This class is used to override the default behavior of the |     This class is used to override the default behavior of the | ||||||
|     HTTPRedirectHandler, which does *not* redirect POST data |     HTTPRedirectHandler, which does *not* redirect POST data | ||||||
| @ -81,24 +69,30 @@ class PysHTTPRedirectHandler(urllib2.HTTPRedirectHandler): | |||||||
|         if (code in (301, 302, 303, 307) and m in ("GET", "HEAD") |         if (code in (301, 302, 303, 307) and m in ("GET", "HEAD") | ||||||
|             or code in (301, 302, 303) and m == "POST"): |             or code in (301, 302, 303) and m == "POST"): | ||||||
|             newurl = newurl.replace(' ', '%20') |             newurl = newurl.replace(' ', '%20') | ||||||
|             newheaders = dict((k, v) for k, v in req.headers.items() |             newheaders = dict((k, v) for k, v in list(req.headers.items()) | ||||||
|                 if k.lower() not in ("content-length", "content-type") |                 if k.lower() not in ("content-length", "content-type") | ||||||
|             ) |             ) | ||||||
|             data = None |             data = None | ||||||
|             if req.has_data(): |             if req.data: | ||||||
|                 data = req.get_data() |                 data = req.data | ||||||
|             return urllib2.Request(newurl, |             return urllib.request.Request(newurl, | ||||||
|                            data=data, |                            data=data, | ||||||
|                            headers=newheaders, |                            headers=newheaders, | ||||||
|                            origin_req_host=req.get_origin_req_host(), |                            origin_req_host=req.origin_req_host, | ||||||
|                            unverifiable=True) |                            unverifiable=True) | ||||||
|         else: |         else: | ||||||
|             raise urllib2.HTTPError(req.get_full_url(), code, msg, headers, fp) |             raise urllib.error.HTTPError( | ||||||
|  |                 req.get_full_url(), | ||||||
|  |                 code, | ||||||
|  |                 msg, | ||||||
|  |                 headers, | ||||||
|  |                 fp, | ||||||
|  |             ) | ||||||
|  |  | ||||||
| class Connection(object): | class Connection(object): | ||||||
|     def __init__(self, baseUrl, username=None, password=None, port=4040, |     def __init__(self, baseUrl, username=None, password=None, port=4040, | ||||||
|             serverPath='/rest', appName='py-sonic', apiVersion=API_VERSION, |             serverPath='/rest', appName='py-sonic', apiVersion=API_VERSION, | ||||||
|             insecure=False, useNetrc=None, legacyAuth=False, useGET=False): |             insecure=False, useNetrc=None, legacyAuth=False, useGET=True): | ||||||
|         """ |         """ | ||||||
|         This will create a connection to your subsonic server |         This will create a connection to your subsonic server | ||||||
|  |  | ||||||
| @ -271,6 +265,52 @@ class Connection(object): | |||||||
|         self._checkStatus(res) |         self._checkStatus(res) | ||||||
|         return res |         return res | ||||||
|  |  | ||||||
|  |     def getScanStatus(self): | ||||||
|  |         """ | ||||||
|  |         since: 1.15.0 | ||||||
|  |  | ||||||
|  |         returns the current status for media library scanning. | ||||||
|  |         takes no extra parameters. | ||||||
|  |  | ||||||
|  |         returns a dict like the following: | ||||||
|  |  | ||||||
|  |         {'status': 'ok', 'version': '1.15.0', | ||||||
|  |         'scanstatus': {'scanning': true, 'count': 4680}} | ||||||
|  |  | ||||||
|  |         'count' is the total number of items to be scanned | ||||||
|  |         """ | ||||||
|  |         methodName = 'getScanStatus' | ||||||
|  |         viewName = '%s.view' % methodName | ||||||
|  |  | ||||||
|  |         req = self._getRequest(viewName) | ||||||
|  |         res = self._doInfoReq(req) | ||||||
|  |         self._checkStatus(res) | ||||||
|  |         return res | ||||||
|  |  | ||||||
|  |     def startScan(self): | ||||||
|  |         """ | ||||||
|  |         since: 1.15.0 | ||||||
|  |  | ||||||
|  |         Initiates a rescan of the media libraries. | ||||||
|  |         Takes no extra parameters. | ||||||
|  |  | ||||||
|  |         returns a dict like the following: | ||||||
|  |  | ||||||
|  |         {'status': 'ok', 'version': '1.15.0', | ||||||
|  |         'scanstatus': {'scanning': true, 'count': 0}} | ||||||
|  |  | ||||||
|  |         'scanning' changes to false when a scan is complete | ||||||
|  |         'count' starts a 0 and ends at the total number of items scanned | ||||||
|  |  | ||||||
|  |         """ | ||||||
|  |         methodName = 'startScan' | ||||||
|  |         viewName = '%s.view' % methodName | ||||||
|  |  | ||||||
|  |         req = self._getRequest(viewName) | ||||||
|  |         res = self._doInfoReq(req) | ||||||
|  |         self._checkStatus(res) | ||||||
|  |         return res | ||||||
|  |  | ||||||
|     def getMusicFolders(self): |     def getMusicFolders(self): | ||||||
|         """ |         """ | ||||||
|         since: 1.0.0 |         since: 1.0.0 | ||||||
| @ -809,6 +849,58 @@ class Connection(object): | |||||||
|             self._checkStatus(res) |             self._checkStatus(res) | ||||||
|         return res |         return res | ||||||
|  |  | ||||||
|  |     def streamUrl(self, sid, maxBitRate=0, tformat=None, timeOffset=None, | ||||||
|  |             size=None, estimateContentLength=False, converted=False): | ||||||
|  |         """ | ||||||
|  |         since: 1.0.0 | ||||||
|  |  | ||||||
|  |         Downloads a given music file. | ||||||
|  |  | ||||||
|  |         sid:str         The ID of the music file to download. | ||||||
|  |         maxBitRate:int  (since: 1.2.0) If specified, the server will | ||||||
|  |                         attempt to limit the bitrate to this value, in | ||||||
|  |                         kilobits per second. If set to zero (default), no limit | ||||||
|  |                         is imposed. Legal values are: 0, 32, 40, 48, 56, 64, | ||||||
|  |                         80, 96, 112, 128, 160, 192, 224, 256 and 320. | ||||||
|  |         tformat:str     (since: 1.6.0) Specifies the target format | ||||||
|  |                         (e.g. "mp3" or "flv") in case there are multiple | ||||||
|  |                         applicable transcodings (since: 1.9.0) You can use | ||||||
|  |                         the special value "raw" to disable transcoding | ||||||
|  |         timeOffset:int  (since: 1.6.0) Only applicable to video | ||||||
|  |                         streaming.  Start the stream at the given | ||||||
|  |                         offset (in seconds) into the video | ||||||
|  |         size:str        (since: 1.6.0) The requested video size in | ||||||
|  |                         WxH, for instance 640x480 | ||||||
|  |         estimateContentLength:bool  (since: 1.8.0) If set to True, | ||||||
|  |                                     the HTTP Content-Length header | ||||||
|  |                                     will be set to an estimated | ||||||
|  |                                     value for trancoded media | ||||||
|  |         converted:bool  (since: 1.14.0) Only applicable to video streaming. | ||||||
|  |                         Subsonic can optimize videos for streaming by | ||||||
|  |                         converting them to MP4. If a conversion exists for | ||||||
|  |                         the video in question, then setting this parameter | ||||||
|  |                         to "true" will cause the converted video to be | ||||||
|  |                         returned instead of the original. | ||||||
|  |  | ||||||
|  |         Returns the file-like object for reading or raises an exception | ||||||
|  |         on error | ||||||
|  |         """ | ||||||
|  |         methodName = 'stream' | ||||||
|  |         viewName = '%s.view' % methodName | ||||||
|  |  | ||||||
|  |         q = self._getQueryDict({'id': sid, 'maxBitRate': maxBitRate, | ||||||
|  |             'format': tformat, 'timeOffset': timeOffset, 'size': size, | ||||||
|  |             'estimateContentLength': estimateContentLength, | ||||||
|  |             'converted': converted}) | ||||||
|  |  | ||||||
|  |         req = self._getRequest(viewName, q) | ||||||
|  |         xbmc.log("Requesting %s"%str(req.full_url),xbmc.LOGDEBUG) | ||||||
|  |         res = self._doBinReq(req) | ||||||
|  |         if isinstance(res, dict): | ||||||
|  |             self._checkStatus(res) | ||||||
|  |         return req.full_url | ||||||
|  |  | ||||||
|  |  | ||||||
|     def getCoverArt(self, aid, size=None): |     def getCoverArt(self, aid, size=None): | ||||||
|         """ |         """ | ||||||
|         since: 1.0.0 |         since: 1.0.0 | ||||||
| @ -832,6 +924,30 @@ class Connection(object): | |||||||
|             self._checkStatus(res) |             self._checkStatus(res) | ||||||
|         return res |         return res | ||||||
|  |  | ||||||
|  |     def getCoverArtUrl(self, aid, size=None): | ||||||
|  |         """ | ||||||
|  |         since: 1.0.0 | ||||||
|  |  | ||||||
|  |         Returns a cover art image | ||||||
|  |  | ||||||
|  |         aid:str     ID string for the cover art image to download | ||||||
|  |         size:int    If specified, scale image to this size | ||||||
|  |  | ||||||
|  |         Returns the file-like object for reading or raises an exception | ||||||
|  |         on error | ||||||
|  |         """ | ||||||
|  |         methodName = 'getCoverArt' | ||||||
|  |         viewName = '%s.view' % methodName | ||||||
|  |  | ||||||
|  |         q = self._getQueryDict({'id': aid, 'size': size}) | ||||||
|  |  | ||||||
|  |         req = self._getRequest(viewName, q) | ||||||
|  |         res = self._doBinReq(req) | ||||||
|  |         if isinstance(res, dict): | ||||||
|  |             self._checkStatus(res) | ||||||
|  |         return req.full_url | ||||||
|  |  | ||||||
|  |  | ||||||
|     def scrobble(self, sid, submission=True, listenTime=None): |     def scrobble(self, sid, submission=True, listenTime=None): | ||||||
|         """ |         """ | ||||||
|         since: 1.5.0 |         since: 1.5.0 | ||||||
| @ -980,7 +1096,7 @@ class Connection(object): | |||||||
|             streamRole=True, jukeboxRole=False, downloadRole=False, |             streamRole=True, jukeboxRole=False, downloadRole=False, | ||||||
|             uploadRole=False, playlistRole=False, coverArtRole=False, |             uploadRole=False, playlistRole=False, coverArtRole=False, | ||||||
|             commentRole=False, podcastRole=False, shareRole=False, |             commentRole=False, podcastRole=False, shareRole=False, | ||||||
|             musicFolderId=None): |             videoConversionRole=False, musicFolderId=None): | ||||||
|         """ |         """ | ||||||
|         since: 1.1.0 |         since: 1.1.0 | ||||||
|  |  | ||||||
| @ -1011,6 +1127,7 @@ class Connection(object): | |||||||
|             'uploadRole': uploadRole, 'playlistRole': playlistRole, |             'uploadRole': uploadRole, 'playlistRole': playlistRole, | ||||||
|             'coverArtRole': coverArtRole, 'commentRole': commentRole, |             'coverArtRole': coverArtRole, 'commentRole': commentRole, | ||||||
|             'podcastRole': podcastRole, 'shareRole': shareRole, |             'podcastRole': podcastRole, 'shareRole': shareRole, | ||||||
|  |             'videoConversionRole': videoConversionRole, | ||||||
|             'musicFolderId': musicFolderId |             'musicFolderId': musicFolderId | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
| @ -1024,7 +1141,7 @@ class Connection(object): | |||||||
|             streamRole=True, jukeboxRole=False, downloadRole=False, |             streamRole=True, jukeboxRole=False, downloadRole=False, | ||||||
|             uploadRole=False, playlistRole=False, coverArtRole=False, |             uploadRole=False, playlistRole=False, coverArtRole=False, | ||||||
|             commentRole=False, podcastRole=False, shareRole=False, |             commentRole=False, podcastRole=False, shareRole=False, | ||||||
|             musicFolderId=None, maxBitRate=0): |             videoConversionRole=False, musicFolderId=None, maxBitRate=0): | ||||||
|         """ |         """ | ||||||
|         since 1.10.1 |         since 1.10.1 | ||||||
|  |  | ||||||
| @ -1056,6 +1173,7 @@ class Connection(object): | |||||||
|             'uploadRole': uploadRole, 'playlistRole': playlistRole, |             'uploadRole': uploadRole, 'playlistRole': playlistRole, | ||||||
|             'coverArtRole': coverArtRole, 'commentRole': commentRole, |             'coverArtRole': coverArtRole, 'commentRole': commentRole, | ||||||
|             'podcastRole': podcastRole, 'shareRole': shareRole, |             'podcastRole': podcastRole, 'shareRole': shareRole, | ||||||
|  |             'videoConversionRole': videoConversionRole, | ||||||
|             'musicFolderId': musicFolderId, 'maxBitRate': maxBitRate |             'musicFolderId': musicFolderId, 'maxBitRate': maxBitRate | ||||||
|         }) |         }) | ||||||
|         req = self._getRequest(viewName, q) |         req = self._getRequest(viewName, q) | ||||||
| @ -1960,7 +2078,7 @@ class Connection(object): | |||||||
|         req = self._getRequest(viewName, q) |         req = self._getRequest(viewName, q) | ||||||
|         try: |         try: | ||||||
|             res = self._doBinReq(req) |             res = self._doBinReq(req) | ||||||
|         except urllib2.HTTPError: |         except urllib.error.HTTPError: | ||||||
|             # Avatar is not set/does not exist, return None |             # Avatar is not set/does not exist, return None | ||||||
|             return None |             return None | ||||||
|         if isinstance(res, dict): |         if isinstance(res, dict): | ||||||
| @ -2065,7 +2183,7 @@ class Connection(object): | |||||||
|         musicFolderId:int   Only return results from the music folder |         musicFolderId:int   Only return results from the music folder | ||||||
|                             with the given ID. See getMusicFolders |                             with the given ID. See getMusicFolders | ||||||
|         """ |         """ | ||||||
|         methodName = 'getGenres' |         methodName = 'getSongsByGenre' | ||||||
|         viewName = '%s.view' % methodName |         viewName = '%s.view' % methodName | ||||||
|  |  | ||||||
|         q = self._getQueryDict({'genre': genre, |         q = self._getQueryDict({'genre': genre, | ||||||
| @ -2112,7 +2230,7 @@ class Connection(object): | |||||||
|         req = self._getRequest(viewName, q) |         req = self._getRequest(viewName, q) | ||||||
|         try: |         try: | ||||||
|             res = self._doBinReq(req) |             res = self._doBinReq(req) | ||||||
|         except urllib2.HTTPError: |         except urllib.error.HTTPError: | ||||||
|             # Avatar is not set/does not exist, return None |             # Avatar is not set/does not exist, return None | ||||||
|             return None |             return None | ||||||
|         if isinstance(res, dict): |         if isinstance(res, dict): | ||||||
| @ -2224,6 +2342,70 @@ class Connection(object): | |||||||
|         self._checkStatus(res) |         self._checkStatus(res) | ||||||
|         return res |         return res | ||||||
|  |  | ||||||
|  |     def createInternetRadioStation(self, streamUrl, name, homepageUrl=None): | ||||||
|  |         """ | ||||||
|  |         since 1.16.0 | ||||||
|  |  | ||||||
|  |         Create an internet radio station | ||||||
|  |  | ||||||
|  |         streamUrl:str   The stream URL for the station | ||||||
|  |         name:str        The user-defined name for the station | ||||||
|  |         homepageUrl:str The homepage URL for the station | ||||||
|  |         """ | ||||||
|  |         methodName = 'createInternetRadioStation' | ||||||
|  |         viewName = '{}.view'.format(methodName) | ||||||
|  |  | ||||||
|  |         q = self._getQueryDict({ | ||||||
|  |             'streamUrl': streamUrl, 'name': name, 'homepageUrl': homepageUrl}) | ||||||
|  |  | ||||||
|  |         req = self._getRequest(viewName, q) | ||||||
|  |         res = self._doInfoReq(req) | ||||||
|  |         self._checkStatus(res) | ||||||
|  |         return res | ||||||
|  |  | ||||||
|  |     def updateInternetRadioStation(self, iid, streamUrl, name, | ||||||
|  |             homepageUrl=None): | ||||||
|  |         """ | ||||||
|  |         since 1.16.0 | ||||||
|  |  | ||||||
|  |         Create an internet radio station | ||||||
|  |  | ||||||
|  |         iid:str         The ID for the station | ||||||
|  |         streamUrl:str   The stream URL for the station | ||||||
|  |         name:str        The user-defined name for the station | ||||||
|  |         homepageUrl:str The homepage URL for the station | ||||||
|  |         """ | ||||||
|  |         methodName = 'updateInternetRadioStation' | ||||||
|  |         viewName = '{}.view'.format(methodName) | ||||||
|  |  | ||||||
|  |         q = self._getQueryDict({ | ||||||
|  |             'id': iid, 'streamUrl': streamUrl, 'name': name, | ||||||
|  |             'homepageUrl': homepageUrl, | ||||||
|  |         }) | ||||||
|  |  | ||||||
|  |         req = self._getRequest(viewName, q) | ||||||
|  |         res = self._doInfoReq(req) | ||||||
|  |         self._checkStatus(res) | ||||||
|  |         return res | ||||||
|  |  | ||||||
|  |     def deleteInternetRadioStation(self, iid): | ||||||
|  |         """ | ||||||
|  |         since 1.16.0 | ||||||
|  |  | ||||||
|  |         Create an internet radio station | ||||||
|  |  | ||||||
|  |         iid:str         The ID for the station | ||||||
|  |         """ | ||||||
|  |         methodName = 'deleteInternetRadioStation' | ||||||
|  |         viewName = '{}.view'.format(methodName) | ||||||
|  |  | ||||||
|  |         q = {'id': iid} | ||||||
|  |  | ||||||
|  |         req = self._getRequest(viewName, q) | ||||||
|  |         res = self._doInfoReq(req) | ||||||
|  |         self._checkStatus(res) | ||||||
|  |         return res | ||||||
|  |  | ||||||
|     def getBookmarks(self): |     def getBookmarks(self): | ||||||
|         """ |         """ | ||||||
|         since: 1.9.0 |         since: 1.9.0 | ||||||
| @ -2561,7 +2743,7 @@ class Connection(object): | |||||||
|  |  | ||||||
|         url = '%s:%d/%s/%s?%s' % (self._baseUrl, self._port, |         url = '%s:%d/%s/%s?%s' % (self._baseUrl, self._port, | ||||||
|             self._separateServerPath(), viewName, methodName) |             self._separateServerPath(), viewName, methodName) | ||||||
|         req = urllib2.Request(url) |         req = urllib.request.Request(url) | ||||||
|         res = self._opener.open(req) |         res = self._opener.open(req) | ||||||
|         res_msg = res.msg.lower() |         res_msg = res.msg.lower() | ||||||
|         return res_msg == 'ok' |         return res_msg == 'ok' | ||||||
| @ -2575,14 +2757,17 @@ class Connection(object): | |||||||
|         if sys.version_info[:3] >= (2, 7, 9) and self._insecure: |         if sys.version_info[:3] >= (2, 7, 9) and self._insecure: | ||||||
|             https_chain = HTTPSHandlerChain( |             https_chain = HTTPSHandlerChain( | ||||||
|                 context=ssl._create_unverified_context()) |                 context=ssl._create_unverified_context()) | ||||||
|         opener = urllib2.build_opener(PysHTTPRedirectHandler, https_chain) |         opener = urllib.request.build_opener( | ||||||
|  |             PysHTTPRedirectHandler, | ||||||
|  |             https_chain, | ||||||
|  |         ) | ||||||
|         return opener |         return opener | ||||||
|  |  | ||||||
|     def _getQueryDict(self, d): |     def _getQueryDict(self, d): | ||||||
|         """ |         """ | ||||||
|         Given a dictionary, it cleans out all the values set to None |         Given a dictionary, it cleans out all the values set to None | ||||||
|         """ |         """ | ||||||
|         for k, v in d.items(): |         for k, v in list(d.items()): | ||||||
|             if v is None: |             if v is None: | ||||||
|                 del d[k] |                 del d[k] | ||||||
|         return d |         return d | ||||||
| @ -2599,7 +2784,7 @@ class Connection(object): | |||||||
|             qdict['p'] = 'enc:%s' % self._hexEnc(self._rawPass) |             qdict['p'] = 'enc:%s' % self._hexEnc(self._rawPass) | ||||||
|         else: |         else: | ||||||
|             salt = self._getSalt() |             salt = self._getSalt() | ||||||
|             token = md5(self._rawPass + salt).hexdigest() |             token = md5((self._rawPass + salt).encode('utf-8')).hexdigest() | ||||||
|             qdict.update({ |             qdict.update({ | ||||||
|                 's': salt, |                 's': salt, | ||||||
|                 't': token, |                 't': token, | ||||||
| @ -2612,12 +2797,13 @@ class Connection(object): | |||||||
|         qdict.update(query) |         qdict.update(query) | ||||||
|         url = '%s:%d/%s/%s' % (self._baseUrl, self._port, self._serverPath, |         url = '%s:%d/%s/%s' % (self._baseUrl, self._port, self._serverPath, | ||||||
|             viewName) |             viewName) | ||||||
|         req = urllib2.Request(url, urlencode(qdict)) |         xbmc.log("Standard URL %s"%url,level=xbmc.LOGDEBUG) | ||||||
|  |         xbmc.log("Qdict %s"%str(qdict),level=xbmc.LOGDEBUG) | ||||||
|         if self._useGET: |         req = urllib.request.Request(url, urlencode(qdict).encode('utf-8')) | ||||||
|  |         if(self._useGET or ('getCoverArt' in viewName) or ('stream' in viewName)): | ||||||
|             url += '?%s' % urlencode(qdict) |             url += '?%s' % urlencode(qdict) | ||||||
|             req = urllib2.Request(url) |             xbmc.log("UseGET URL %s"%(url),xbmc.LOGDEBUG) | ||||||
|  |             req = urllib.request.Request(url) | ||||||
|         return req |         return req | ||||||
|  |  | ||||||
|     def _getRequestWithList(self, viewName, listName, alist, query={}): |     def _getRequestWithList(self, viewName, listName, alist, query={}): | ||||||
| @ -2633,7 +2819,7 @@ class Connection(object): | |||||||
|         data.write(urlencode(qdict)) |         data.write(urlencode(qdict)) | ||||||
|         for i in alist: |         for i in alist: | ||||||
|             data.write('&%s' % urlencode({listName: i})) |             data.write('&%s' % urlencode({listName: i})) | ||||||
|         req = urllib2.Request(url, data.getvalue()) |         req = urllib.request.Request(url, data.getvalue().encode('utf-8')) | ||||||
|  |  | ||||||
|         if self._useGET: |         if self._useGET: | ||||||
|             url += '?%s' % data.getvalue() |             url += '?%s' % data.getvalue() | ||||||
| @ -2657,10 +2843,10 @@ class Connection(object): | |||||||
|             viewName) |             viewName) | ||||||
|         data = StringIO() |         data = StringIO() | ||||||
|         data.write(urlencode(qdict)) |         data.write(urlencode(qdict)) | ||||||
|         for k, l in listMap.iteritems(): |         for k, l in listMap.items(): | ||||||
|             for i in l: |             for i in l: | ||||||
|                 data.write('&%s' % urlencode({k: i})) |                 data.write('&%s' % urlencode({k: i})) | ||||||
|         req = urllib2.Request(url, data.getvalue()) |         req = urllib.request.Request(url, data.getvalue().encode('utf-8')) | ||||||
|  |  | ||||||
|         if self._useGET: |         if self._useGET: | ||||||
|             url += '?%s' % data.getvalue() |             url += '?%s' % data.getvalue() | ||||||
| @ -2671,12 +2857,17 @@ class Connection(object): | |||||||
|     def _doInfoReq(self, req): |     def _doInfoReq(self, req): | ||||||
|         # Returns a parsed dictionary version of the result |         # Returns a parsed dictionary version of the result | ||||||
|         res = self._opener.open(req) |         res = self._opener.open(req) | ||||||
|         dres = json.loads(res.read()) |         dres = json.loads(res.read().decode('utf-8')) | ||||||
|         return dres['subsonic-response'] |         return dres['subsonic-response'] | ||||||
|  |  | ||||||
|     def _doBinReq(self, req): |     def _doBinReq(self, req): | ||||||
|         res = self._opener.open(req) |         res = self._opener.open(req) | ||||||
|         contType = res.info().getheader('Content-Type') |         info = res.info() | ||||||
|  |         if hasattr(info, 'getheader'): | ||||||
|  |             contType = info.getheader('Content-Type') | ||||||
|  |         else: | ||||||
|  |             contType = info.get('Content-Type') | ||||||
|  |  | ||||||
|         if contType: |         if contType: | ||||||
|             if contType.startswith('text/html') or \ |             if contType.startswith('text/html') or \ | ||||||
|                     contType.startswith('application/json'): |                     contType.startswith('application/json'): | ||||||
| @ -2716,7 +2907,7 @@ class Connection(object): | |||||||
|         """ |         """ | ||||||
|         separate REST portion of URL from base server path. |         separate REST portion of URL from base server path. | ||||||
|         """ |         """ | ||||||
|         return urllib2.splithost(self._serverPath)[1].split('/')[0] |         return urllib.parse.splithost(self._serverPath)[1].split('/')[0] | ||||||
|  |  | ||||||
|     def _fixLastModified(self, data): |     def _fixLastModified(self, data): | ||||||
|         """ |         """ | ||||||
| @ -2726,9 +2917,9 @@ class Connection(object): | |||||||
|         of SECONDS since the unix epoch.  JAVA SUCKS! |         of SECONDS since the unix epoch.  JAVA SUCKS! | ||||||
|         """ |         """ | ||||||
|         if isinstance(data, dict): |         if isinstance(data, dict): | ||||||
|             for k, v in data.items(): |             for k, v in list(data.items()): | ||||||
|                 if k == 'lastModified': |                 if k == 'lastModified': | ||||||
|                     data[k] = long(v) / 1000.0 |                     data[k] = int(v) / 1000.0 | ||||||
|                     return |                     return | ||||||
|                 elif isinstance(v, (tuple, list, dict)): |                 elif isinstance(v, (tuple, list, dict)): | ||||||
|                     return self._fixLastModified(v) |                     return self._fixLastModified(v) | ||||||
|  | |||||||
| @ -1,442 +0,0 @@ | |||||||
| import urllib |  | ||||||
| import urlparse |  | ||||||
| import libsonic |  | ||||||
|  |  | ||||||
| def force_list(value): |  | ||||||
|     """ |  | ||||||
|     Coerce the input value to a list. |  | ||||||
|  |  | ||||||
|     If `value` is `None`, return an empty list. If it is a single value, create |  | ||||||
|     a new list with that element on index 0. |  | ||||||
|  |  | ||||||
|     :param value: Input value to coerce. |  | ||||||
|     :return: Value as list. |  | ||||||
|     :rtype: list |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     if value is None: |  | ||||||
|         return [] |  | ||||||
|     elif type(value) == list: |  | ||||||
|         return value |  | ||||||
|     else: |  | ||||||
|         return [value] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class SubsonicClient(libsonic.Connection): |  | ||||||
|     """ |  | ||||||
|     Extend `libsonic.Connection` with new features and fix a few issues. |  | ||||||
|  |  | ||||||
|     - Parse URL for host and port for constructor. |  | ||||||
|     - Make sure API results are of of uniform type. |  | ||||||
|     - Provide methods to intercept URL of binary requests. |  | ||||||
|     - Add order property to playlist items. |  | ||||||
|     - Add conventient `walk_*' methods to iterate over the API responses. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     def __init__(self, url, username, password, apiversion, insecure, legacyauth): |  | ||||||
|         """ |  | ||||||
|         Construct a new SubsonicClient. |  | ||||||
|  |  | ||||||
|         :param str url: Full URL (including scheme) of the Subsonic server. |  | ||||||
|         :param str username: Username of the server. |  | ||||||
|         :param str password: Password of the server. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         self.intercept_url = False |  | ||||||
|  |  | ||||||
|         # Parse Subsonic URL |  | ||||||
|         parts = urlparse.urlparse(url) |  | ||||||
|         scheme = parts.scheme or "http" |  | ||||||
|  |  | ||||||
|         # Make sure there is hostname |  | ||||||
|         if not parts.hostname: |  | ||||||
|             raise ValueError("Expected hostname for URL: %s" % url) |  | ||||||
|  |  | ||||||
|         # Validate scheme |  | ||||||
|         if scheme not in ("http", "https"): |  | ||||||
|             raise ValueError("Unexpected scheme '%s' for URL: %s" % ( |  | ||||||
|                 scheme, url)) |  | ||||||
|  |  | ||||||
|         # Pick a default port |  | ||||||
|         host = "%s://%s" % (scheme, parts.hostname) |  | ||||||
|         port = parts.port or {"http": 80, "https": 443}[scheme] |  | ||||||
|         path = parts.path.rstrip('/') + '/rest' |  | ||||||
|  |  | ||||||
|         # Invoke original constructor |  | ||||||
|         super(SubsonicClient, self).__init__( |  | ||||||
|             host, username, password, port=port, serverPath=path, appName='Kodi', apiVersion=apiversion, insecure=insecure, legacyAuth=legacyauth) |  | ||||||
|  |  | ||||||
|     def getIndexes(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         Improve the getIndexes method. Ensures IDs are integers. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         def _artists_iterator(artists): |  | ||||||
|             for artist in force_list(artists): |  | ||||||
|                 artist["id"] = int(artist["id"]) |  | ||||||
|                 yield artist |  | ||||||
|  |  | ||||||
|         def _index_iterator(index): |  | ||||||
|             for index in force_list(index): |  | ||||||
|                 index["artist"] = list(_artists_iterator(index.get("artist"))) |  | ||||||
|                 yield index |  | ||||||
|  |  | ||||||
|         def _children_iterator(children): |  | ||||||
|             for child in force_list(children): |  | ||||||
|                 child["id"] = int(child["id"]) |  | ||||||
|  |  | ||||||
|                 if "parent" in child: |  | ||||||
|                     child["parent"] = int(child["parent"]) |  | ||||||
|                 if "coverArt" in child: |  | ||||||
|                     child["coverArt"] = int(child["coverArt"]) |  | ||||||
|                 if "artistId" in child: |  | ||||||
|                     child["artistId"] = int(child["artistId"]) |  | ||||||
|                 if "albumId" in child: |  | ||||||
|                     child["albumId"] = int(child["albumId"]) |  | ||||||
|  |  | ||||||
|                 yield child |  | ||||||
|  |  | ||||||
|         response = super(SubsonicClient, self).getIndexes(*args, **kwargs) |  | ||||||
|         response["indexes"] = response.get("indexes", {}) |  | ||||||
|         response["indexes"]["index"] = list( |  | ||||||
|             _index_iterator(response["indexes"].get("index"))) |  | ||||||
|         response["indexes"]["child"] = list( |  | ||||||
|             _children_iterator(response["indexes"].get("child"))) |  | ||||||
|  |  | ||||||
|         return response |  | ||||||
|  |  | ||||||
|     def getPlaylists(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         Improve the getPlaylists method. Ensures IDs are integers. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         def _playlists_iterator(playlists): |  | ||||||
|             for playlist in force_list(playlists): |  | ||||||
|                 playlist["id"] = int(playlist["id"]) |  | ||||||
|                 yield playlist |  | ||||||
|  |  | ||||||
|         response = super(SubsonicClient, self).getPlaylists(*args, **kwargs) |  | ||||||
|         response["playlists"]["playlist"] = list( |  | ||||||
|             _playlists_iterator(response["playlists"].get("playlist"))) |  | ||||||
|  |  | ||||||
|         return response |  | ||||||
|  |  | ||||||
|     def getPlaylist(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         Improve the getPlaylist method. Ensures IDs are integers and add an |  | ||||||
|         order property to each entry. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         def _entries_iterator(entries): |  | ||||||
|             for order, entry in enumerate(force_list(entries), start=1): |  | ||||||
|                 entry["id"] = int(entry["id"]) |  | ||||||
|                 entry["order"] = order |  | ||||||
|                 yield entry |  | ||||||
|  |  | ||||||
|         response = super(SubsonicClient, self).getPlaylist(*args, **kwargs) |  | ||||||
|         response["playlist"]["entry"] = list( |  | ||||||
|             _entries_iterator(response["playlist"].get("entry"))) |  | ||||||
|  |  | ||||||
|         return response |  | ||||||
|  |  | ||||||
|     def getArtists(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         (ID3 tags) |  | ||||||
|         Improve the getArtists method. Ensures IDs are integers. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         def _artists_iterator(artists): |  | ||||||
|             for artist in force_list(artists): |  | ||||||
|                 artist["id"] = int(artist["id"]) |  | ||||||
|                 yield artist |  | ||||||
|  |  | ||||||
|         def _index_iterator(index): |  | ||||||
|             for index in force_list(index): |  | ||||||
|                 index["artist"] = list(_artists_iterator(index.get("artist"))) |  | ||||||
|                 yield index |  | ||||||
|  |  | ||||||
|         response = super(SubsonicClient, self).getArtists(*args, **kwargs) |  | ||||||
|         response["artists"] = response.get("artists", {}) |  | ||||||
|         response["artists"]["index"] = list( |  | ||||||
|             _index_iterator(response["artists"].get("index"))) |  | ||||||
|  |  | ||||||
|         return response |  | ||||||
|  |  | ||||||
|     def getArtist(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         (ID3 tags)  |  | ||||||
|         Improve the getArtist method. Ensures IDs are integers. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         def _albums_iterator(albums): |  | ||||||
|             for album in force_list(albums): |  | ||||||
|                 album["id"] = int(album["id"]) |  | ||||||
|  |  | ||||||
|                 if "artistId" in album: |  | ||||||
|                     album["artistId"] = int(album["artistId"]) |  | ||||||
|  |  | ||||||
|                 yield album |  | ||||||
|  |  | ||||||
|         response = super(SubsonicClient, self).getArtist(*args, **kwargs) |  | ||||||
|         response["artist"]["album"] = list( |  | ||||||
|             _albums_iterator(response["artist"].get("album"))) |  | ||||||
|  |  | ||||||
|         return response |  | ||||||
|  |  | ||||||
|     def getMusicDirectory(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         Improve the getMusicDirectory method. Ensures IDs are integers. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         def _children_iterator(children): |  | ||||||
|             for child in force_list(children): |  | ||||||
|                 child["id"] = int(child["id"]) |  | ||||||
|  |  | ||||||
|                 if "parent" in child: |  | ||||||
|                     child["parent"] = int(child["parent"]) |  | ||||||
|                 if "coverArt" in child: |  | ||||||
|                     child["coverArt"] = int(child["coverArt"]) |  | ||||||
|                 if "artistId" in child: |  | ||||||
|                     child["artistId"] = int(child["artistId"]) |  | ||||||
|                 if "albumId" in child: |  | ||||||
|                     child["albumId"] = int(child["albumId"]) |  | ||||||
|  |  | ||||||
|                 yield child |  | ||||||
|  |  | ||||||
|         response = super(SubsonicClient, self).getMusicDirectory( |  | ||||||
|             *args, **kwargs) |  | ||||||
|         response["directory"]["child"] = list( |  | ||||||
|             _children_iterator(response["directory"].get("child"))) |  | ||||||
|  |  | ||||||
|         return response |  | ||||||
|  |  | ||||||
|     def getAlbum(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         (ID3 tags) |  | ||||||
|         Improve the getAlbum method. Ensures the IDs are real integers. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         def _songs_iterator(songs): |  | ||||||
|             for song in force_list(songs): |  | ||||||
|                 song["id"] = int(song["id"]) |  | ||||||
|                 yield song |  | ||||||
|  |  | ||||||
|         response = super(SubsonicClient, self).getAlbum(*args, **kwargs) |  | ||||||
|         response["album"]["song"] = list( |  | ||||||
|             _songs_iterator(response["album"].get("song"))) |  | ||||||
|  |  | ||||||
|         return response |  | ||||||
|  |  | ||||||
|     def getAlbumList2(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         Improve the getAlbumList2 method. Ensures the IDs are real integers. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         def _album_iterator(albums): |  | ||||||
|             for album in force_list(albums): |  | ||||||
|                 album["id"] = int(album["id"]) |  | ||||||
|                 yield album |  | ||||||
|  |  | ||||||
|         response = super(SubsonicClient, self).getAlbumList2(*args, **kwargs) |  | ||||||
|         response["albumList2"]["album"] = list( |  | ||||||
|             _album_iterator(response["albumList2"].get("album"))) |  | ||||||
|  |  | ||||||
|         return response |  | ||||||
|  |  | ||||||
|     def getStarred(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         Improve the getStarred method. Ensures the IDs are real integers. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         def _song_iterator(songs): |  | ||||||
|             for song in force_list(songs): |  | ||||||
|                 song["id"] = int(song["id"]) |  | ||||||
|                 yield song |  | ||||||
|  |  | ||||||
|         response = super(SubsonicClient, self).getStarred(*args, **kwargs) |  | ||||||
|         response["starred"]["song"] = list( |  | ||||||
|             _song_iterator(response["starred"].get("song"))) |  | ||||||
|  |  | ||||||
|         return response |  | ||||||
|  |  | ||||||
|     def getCoverArtUrl(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         Return an URL to the cover art. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         self.intercept_url = True |  | ||||||
|         url = self.getCoverArt(*args, **kwargs) |  | ||||||
|         self.intercept_url = False |  | ||||||
|  |  | ||||||
|         return url |  | ||||||
|  |  | ||||||
|     def streamUrl(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         Return an URL to the file to stream. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         self.intercept_url = True |  | ||||||
|         url = self.stream(*args, **kwargs) |  | ||||||
|         self.intercept_url = False |  | ||||||
|  |  | ||||||
|         return url |  | ||||||
|  |  | ||||||
|     def _doBinReq(self, *args, **kwargs): |  | ||||||
|         """ |  | ||||||
|         Intercept request URL to provide the URL of the item that is requested. |  | ||||||
|  |  | ||||||
|         If the URL is intercepted, the request is not executed. A username and |  | ||||||
|         password is added to provide direct access to the stream. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         if self.intercept_url: |  | ||||||
|             parts = list(urlparse.urlparse( |  | ||||||
|                 args[0].get_full_url() + "?" + args[0].data)) |  | ||||||
|             parts[4] = dict(urlparse.parse_qsl(parts[4])) |  | ||||||
|             if self._legacyAuth: |  | ||||||
|                 parts[4].update({"u": self.username, "p": 'enc:%s' % self._hexEnc(self._rawPass)}) |  | ||||||
|             parts[4] = urllib.urlencode(parts[4]) |  | ||||||
|  |  | ||||||
|             return urlparse.urlunparse(parts) |  | ||||||
|         else: |  | ||||||
|             return super(SubsonicClient, self)._doBinReq(*args, **kwargs) |  | ||||||
|  |  | ||||||
|     def walk_index(self, folder_id=None): |  | ||||||
|         """ |  | ||||||
|         Request Subsonic's index and iterate each item. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         response = self.getIndexes(folder_id) |  | ||||||
|  |  | ||||||
|         for index in response["indexes"]["index"]: |  | ||||||
|             for artist in index["artist"]: |  | ||||||
|                 yield artist |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     def walk_playlists(self): |  | ||||||
|         """ |  | ||||||
|         Request Subsonic's playlists and iterate over each item. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         response = self.getPlaylists() |  | ||||||
|  |  | ||||||
|         for child in response["playlists"]["playlist"]: |  | ||||||
|             yield child |  | ||||||
|  |  | ||||||
|     def walk_playlist(self, playlist_id): |  | ||||||
|         """ |  | ||||||
|         Request Subsonic's playlist items and iterate over each item. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         response = self.getPlaylist(playlist_id) |  | ||||||
|  |  | ||||||
|         for child in response["playlist"]["entry"]: |  | ||||||
|             yield child |  | ||||||
|              |  | ||||||
|     def walk_folders(self): |  | ||||||
|         response = self.getMusicFolders() |  | ||||||
|          |  | ||||||
|         for child in response["musicFolders"]["musicFolder"]: |  | ||||||
|             yield child |  | ||||||
|  |  | ||||||
|     def walk_directory(self, directory_id): |  | ||||||
|         """ |  | ||||||
|         Request a Subsonic music directory and iterate over each item. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         response = self.getMusicDirectory(directory_id) |  | ||||||
|  |  | ||||||
|         for child in response["directory"]["child"]: |  | ||||||
|             if child.get("isDir"): |  | ||||||
|                 for child in self.walk_directory(child["id"]): |  | ||||||
|                     yield child |  | ||||||
|             else: |  | ||||||
|                 yield child |  | ||||||
|  |  | ||||||
|     def walk_artist(self, artist_id): |  | ||||||
|         """ |  | ||||||
|         Request a Subsonic artist and iterate over each album. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         response = self.getArtist(artist_id) |  | ||||||
|  |  | ||||||
|         for child in response["artist"]["album"]: |  | ||||||
|             yield child |  | ||||||
|  |  | ||||||
|     def walk_artists(self): |  | ||||||
|         """ |  | ||||||
|         (ID3 tags) |  | ||||||
|         Request all artists and iterate over each item. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         response = self.getArtists() |  | ||||||
|  |  | ||||||
|         for index in response["artists"]["index"]: |  | ||||||
|             for artist in index["artist"]: |  | ||||||
|                 yield artist |  | ||||||
|  |  | ||||||
|     def walk_genres(self): |  | ||||||
|         """ |  | ||||||
|         (ID3 tags) |  | ||||||
|         Request all genres and iterate over each item. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         response = self.getGenres() |  | ||||||
|  |  | ||||||
|         for genre in response["genres"]["genre"]: |  | ||||||
|             yield genre |  | ||||||
|  |  | ||||||
|     def walk_albums(self, ltype, size=None, fromYear=None,toYear=None, genre=None, offset=None): |  | ||||||
|         """ |  | ||||||
|         (ID3 tags) |  | ||||||
|         Request all albums for a given genre and iterate over each album. |  | ||||||
|         """ |  | ||||||
|          |  | ||||||
|         if ltype == 'byGenre' and genre is None: |  | ||||||
|             return |  | ||||||
|          |  | ||||||
|         if ltype == 'byYear' and (fromYear is None or toYear is None): |  | ||||||
|             return |  | ||||||
|  |  | ||||||
|         response = self.getAlbumList2( |  | ||||||
|             ltype=ltype, size=size, fromYear=fromYear, toYear=toYear,genre=genre, offset=offset) |  | ||||||
|  |  | ||||||
|         if not response["albumList2"]["album"]: |  | ||||||
|             return |  | ||||||
|  |  | ||||||
|         for album in response["albumList2"]["album"]: |  | ||||||
|             yield album |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     def walk_album(self, album_id): |  | ||||||
|         """ |  | ||||||
|         (ID3 tags) |  | ||||||
|         Request an album and iterate over each item. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         response = self.getAlbum(album_id) |  | ||||||
|  |  | ||||||
|         for song in response["album"]["song"]: |  | ||||||
|             yield song |  | ||||||
|  |  | ||||||
|     def walk_tracks_random(self, size=None, genre=None, fromYear=None,toYear=None): |  | ||||||
|         """ |  | ||||||
|         Request random songs by genre and/or year and iterate over each song. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         response = self.getRandomSongs( |  | ||||||
|             size=size, genre=genre, fromYear=fromYear, toYear=toYear) |  | ||||||
|  |  | ||||||
|         for song in response["randomSongs"]["song"]: |  | ||||||
|             yield song |  | ||||||
|              |  | ||||||
|  |  | ||||||
|     def walk_tracks_starred(self): |  | ||||||
|         """ |  | ||||||
|         Request Subsonic's starred songs and iterate over each item. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         response = self.getStarred() |  | ||||||
|  |  | ||||||
|         for song in response["starred"]["song"]: |  | ||||||
|             yield song |  | ||||||
| @ -1,4 +1,4 @@ | |||||||
| #v2.1.0 | #v2.1.0 | ||||||
| #https://github.com/romanvm/script.module.simpleplugin/releases | #https://github.com/romanvm/script.module.simpleplugin/releases | ||||||
|  |  | ||||||
| from simpleplugin import * | from .simpleplugin import * | ||||||
|  | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										484
									
								
								main.py
									
									
									
									
									
								
							
							
						
						
									
										484
									
								
								main.py
									
									
									
									
									
								
							| @ -6,6 +6,7 @@ | |||||||
| # Created on: 14 January 2017 | # Created on: 14 January 2017 | ||||||
| # License: GPL v.3 https://www.gnu.org/copyleft/gpl.html | # License: GPL v.3 https://www.gnu.org/copyleft/gpl.html | ||||||
|  |  | ||||||
|  | import xbmcvfs | ||||||
| import os | import os | ||||||
| import xbmcaddon | import xbmcaddon | ||||||
| import xbmcplugin | import xbmcplugin | ||||||
| @ -14,12 +15,13 @@ import json | |||||||
| import shutil | import shutil | ||||||
| import dateutil.parser | import dateutil.parser | ||||||
| from datetime import datetime | from datetime import datetime | ||||||
|  | from collections import MutableMapping, namedtuple | ||||||
|  |  | ||||||
| # Add the /lib folder to sys | # Add the /lib folder to sys | ||||||
| sys.path.append(xbmc.translatePath(os.path.join(xbmcaddon.Addon("plugin.audio.subsonic").getAddonInfo("path"), "lib"))) | sys.path.append(xbmcvfs.translatePath(os.path.join(xbmcaddon.Addon("plugin.audio.subsonic").getAddonInfo("path"), "lib"))) | ||||||
|  |  | ||||||
|  |  | ||||||
| import libsonic_extra #TO FIX - we should get rid of this and use only libsonic | import libsonic#Removed libsonic_extra | ||||||
|  |  | ||||||
| from simpleplugin import Plugin | from simpleplugin import Plugin | ||||||
| from simpleplugin import Addon | from simpleplugin import Addon | ||||||
| @ -32,7 +34,10 @@ plugin = Plugin() | |||||||
|  |  | ||||||
| connection = None | connection = None | ||||||
| cachetime = int(Addon().get_setting('cachetime')) | cachetime = int(Addon().get_setting('cachetime')) | ||||||
|  | local_starred = set({}) | ||||||
|  |  | ||||||
|  | ListContext = namedtuple('ListContext', ['listing', 'succeeded','update_listing', 'cache_to_disk','sort_methods', 'view_mode','content', 'category']) | ||||||
|  | PlayContext = namedtuple('PlayContext', ['path', 'play_item', 'succeeded']) | ||||||
| def popup(text, time=5000, image=None): | def popup(text, time=5000, image=None): | ||||||
|     title = plugin.addon.getAddonInfo('name') |     title = plugin.addon.getAddonInfo('name') | ||||||
|     icon = plugin.addon.getAddonInfo('icon') |     icon = plugin.addon.getAddonInfo('icon') | ||||||
| @ -42,26 +47,25 @@ def popup(text, time=5000, image=None): | |||||||
| def get_connection(): | def get_connection(): | ||||||
|     global connection |     global connection | ||||||
|      |      | ||||||
|     if connection is None: |     if connection==None:    | ||||||
|          |  | ||||||
|         connected = False   |         connected = False   | ||||||
|          |  | ||||||
|         # Create connection       |         # Create connection       | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             connection = libsonic_extra.SubsonicClient( |             connection = libsonic.Connection( | ||||||
|                 Addon().get_setting('subsonic_url'), |                 baseUrl=Addon().get_setting('subsonic_url'), | ||||||
|                 Addon().get_setting('username', convert=False), |                 username=Addon().get_setting('username', convert=False), | ||||||
|                 Addon().get_setting('password', convert=False), |                 password=Addon().get_setting('password', convert=False), | ||||||
|                 Addon().get_setting('apiversion'), |                 port=Addon().get_setting('port'), | ||||||
|                 Addon().get_setting('insecure') == 'true', |                 apiVersion=Addon().get_setting('apiversion'), | ||||||
|                 Addon().get_setting('legacyauth') == 'true', |                 insecure=Addon().get_setting('insecure'), | ||||||
|  |                 legacyAuth=Addon().get_setting('legacyauth'), | ||||||
|  |                 useGET=Addon().get_setting('useget'), | ||||||
|             )             |             )             | ||||||
|             connected = connection.ping() |             connected = connection.ping() | ||||||
|         except: |         except: | ||||||
|             pass |             pass | ||||||
|  |  | ||||||
|         if connected is False: |         if connected==False: | ||||||
|             popup('Connection error') |             popup('Connection error') | ||||||
|             return False |             return False | ||||||
|  |  | ||||||
| @ -73,7 +77,7 @@ def root(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|      |      | ||||||
|     listing = [] |     listing = [] | ||||||
| @ -129,7 +133,7 @@ def root(params): | |||||||
|                         ) |                         ) | ||||||
|         })  # Item label |         })  # Item label | ||||||
|  |  | ||||||
|     return plugin.create_listing( |     add_directory_items(create_listing( | ||||||
|         listing, |         listing, | ||||||
|         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. |         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. | ||||||
|         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  |         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  | ||||||
| @ -137,7 +141,7 @@ def root(params): | |||||||
|         sort_methods = None, #he list of integer constants representing virtual folder sort methods. |         sort_methods = None, #he list of integer constants representing virtual folder sort methods. | ||||||
|         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). |         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). | ||||||
|         #content = None #string - current plugin content, e.g. ‘movies’ or ‘episodes’. |         #content = None #string - current plugin content, e.g. ‘movies’ or ‘episodes’. | ||||||
|     ) |     )) | ||||||
|  |  | ||||||
| @plugin.action() | @plugin.action() | ||||||
| def menu_albums(params): | def menu_albums(params): | ||||||
| @ -145,7 +149,7 @@ def menu_albums(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|      |      | ||||||
|     listing = [] |     listing = [] | ||||||
| @ -195,7 +199,7 @@ def menu_albums(params): | |||||||
|                         ) |                         ) | ||||||
|         })  # Item label |         })  # Item label | ||||||
|  |  | ||||||
|     return plugin.create_listing( |     add_directory_items(create_listing( | ||||||
|         listing, |         listing, | ||||||
|         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. |         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. | ||||||
|         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  |         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  | ||||||
| @ -203,7 +207,7 @@ def menu_albums(params): | |||||||
|         #sort_methods = None, #he list of integer constants representing virtual folder sort methods. |         #sort_methods = None, #he list of integer constants representing virtual folder sort methods. | ||||||
|         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). |         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). | ||||||
|         #content = None #string - current plugin content, e.g. ‘movies’ or ‘episodes’. |         #content = None #string - current plugin content, e.g. ‘movies’ or ‘episodes’. | ||||||
|     ) |     )) | ||||||
|  |  | ||||||
| @plugin.action() | @plugin.action() | ||||||
| def menu_tracks(params): | def menu_tracks(params): | ||||||
| @ -211,7 +215,7 @@ def menu_tracks(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|      |      | ||||||
|     listing = [] |     listing = [] | ||||||
| @ -247,7 +251,7 @@ def menu_tracks(params): | |||||||
|             ) |             ) | ||||||
|         })  # Item label |         })  # Item label | ||||||
|  |  | ||||||
|     return plugin.create_listing( |     add_directory_items(create_listing( | ||||||
|         listing, |         listing, | ||||||
|         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. |         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. | ||||||
|         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  |         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  | ||||||
| @ -255,7 +259,7 @@ def menu_tracks(params): | |||||||
|         #sort_methods = None, #he list of integer constants representing virtual folder sort methods. |         #sort_methods = None, #he list of integer constants representing virtual folder sort methods. | ||||||
|         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). |         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). | ||||||
|         #content = None #string - current plugin content, e.g. ‘movies’ or ‘episodes’. |         #content = None #string - current plugin content, e.g. ‘movies’ or ‘episodes’. | ||||||
|     ) |     )) | ||||||
|  |  | ||||||
| @plugin.action() | @plugin.action() | ||||||
| #@plugin.cached(cachetime) # cache (in minutes) | #@plugin.cached(cachetime) # cache (in minutes) | ||||||
| @ -263,13 +267,13 @@ def browse_folders(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     listing = [] |     listing = [] | ||||||
|  |  | ||||||
|     # Get items |     # Get items | ||||||
|     items = connection.walk_folders() |     items = walk_folders() | ||||||
|  |  | ||||||
|     # Iterate through items |     # Iterate through items | ||||||
|     for item in items: |     for item in items: | ||||||
| @ -288,7 +292,7 @@ def browse_folders(params): | |||||||
|         plugin.log('One single Media Folder found; do return listing from browse_indexes()...') |         plugin.log('One single Media Folder found; do return listing from browse_indexes()...') | ||||||
|         return browse_indexes(params) |         return browse_indexes(params) | ||||||
|     else: |     else: | ||||||
|         return plugin.create_listing(listing) |         add_directory_items(create_listing(listing)) | ||||||
|  |  | ||||||
| @plugin.action() | @plugin.action() | ||||||
| #@plugin.cached(cachetime) # cache (in minutes) | #@plugin.cached(cachetime) # cache (in minutes) | ||||||
| @ -296,7 +300,7 @@ def browse_indexes(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     listing = [] |     listing = [] | ||||||
| @ -304,7 +308,7 @@ def browse_indexes(params): | |||||||
|     # Get items  |     # Get items  | ||||||
|     # optional folder ID |     # optional folder ID | ||||||
|     folder_id = params.get('folder_id') |     folder_id = params.get('folder_id') | ||||||
|     items = connection.walk_index(folder_id) |     items = walk_index(folder_id) | ||||||
|  |  | ||||||
|     # Iterate through items |     # Iterate through items | ||||||
|     for item in items: |     for item in items: | ||||||
| @ -318,9 +322,9 @@ def browse_indexes(params): | |||||||
|         } |         } | ||||||
|         listing.append(entry) |         listing.append(entry) | ||||||
|          |          | ||||||
|     return plugin.create_listing( |     add_directory_items(create_listing( | ||||||
|         listing |         listing | ||||||
|     ) |     )) | ||||||
|  |  | ||||||
| @plugin.action() | @plugin.action() | ||||||
| #@plugin.cached(cachetime) # cache (in minutes) | #@plugin.cached(cachetime) # cache (in minutes) | ||||||
| @ -328,14 +332,14 @@ def list_directory(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     listing = [] |     listing = [] | ||||||
|      |      | ||||||
|     # Get items |     # Get items | ||||||
|     id = params.get('id') |     id = params.get('id') | ||||||
|     items = connection.walk_directory(id) |     items = walk_directory(id) | ||||||
|  |  | ||||||
|     # Iterate through items |     # Iterate through items | ||||||
|     for item in items: |     for item in items: | ||||||
| @ -356,9 +360,9 @@ def list_directory(params): | |||||||
|  |  | ||||||
|         listing.append(entry) |         listing.append(entry) | ||||||
|          |          | ||||||
|     return plugin.create_listing( |     add_directory_items(create_listing( | ||||||
|         listing |         listing | ||||||
|     ) |     )) | ||||||
|  |  | ||||||
| @plugin.action() | @plugin.action() | ||||||
| #@plugin.cached(cachetime) # cache (in minutes) | #@plugin.cached(cachetime) # cache (in minutes) | ||||||
| @ -370,13 +374,13 @@ def browse_library(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     listing = [] |     listing = [] | ||||||
|  |  | ||||||
|     # Get items |     # Get items | ||||||
|     items = connection.walk_artists() |     items = walk_artists() | ||||||
|  |  | ||||||
|     # Iterate through items |     # Iterate through items | ||||||
|  |  | ||||||
| @ -394,7 +398,7 @@ def browse_library(params): | |||||||
|          |          | ||||||
|         listing.append(entry) |         listing.append(entry) | ||||||
|   |   | ||||||
|     return plugin.create_listing( |     add_directory_items(create_listing( | ||||||
|         listing, |         listing, | ||||||
|         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. |         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. | ||||||
|         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  |         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  | ||||||
| @ -402,7 +406,7 @@ def browse_library(params): | |||||||
|         sort_methods = get_sort_methods('artists',params), #he list of integer constants representing virtual folder sort methods. |         sort_methods = get_sort_methods('artists',params), #he list of integer constants representing virtual folder sort methods. | ||||||
|         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). |         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). | ||||||
|         content = 'artists' #string - current plugin content, e.g. ‘movies’ or ‘episodes’. |         content = 'artists' #string - current plugin content, e.g. ‘movies’ or ‘episodes’. | ||||||
|     ) |     )) | ||||||
|  |  | ||||||
| @plugin.action() | @plugin.action() | ||||||
| #@plugin.cached(cachetime) #cache (in minutes) | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
| @ -417,7 +421,7 @@ def list_albums(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     #query |     #query | ||||||
| @ -443,14 +447,14 @@ def list_albums(params): | |||||||
|  |  | ||||||
|     #Get items |     #Get items | ||||||
|     if 'artist_id' in params: |     if 'artist_id' in params: | ||||||
|         generator = connection.walk_artist(params.get('artist_id')) |         generator = walk_artist(params.get('artist_id')) | ||||||
|     else: |     else: | ||||||
|         generator = connection.walk_albums(**query_args) |         generator = walk_albums(**query_args) | ||||||
|      |      | ||||||
|     #make a list out of the generator so we can iterate it several times |     #make a list out of the generator so we can iterate it several times | ||||||
|     items = list(generator) |     items = list(generator) | ||||||
|      |      | ||||||
|     #check if there is only one artist for this album (and then hide it) |     #check if there==only one artist for this album (and then hide it) | ||||||
|     artists = [item.get('artist',None) for item in items] |     artists = [item.get('artist',None) for item in items] | ||||||
|     if len(artists) <= 1: |     if len(artists) <= 1: | ||||||
|         params['hide_artist']   = True |         params['hide_artist']   = True | ||||||
| @ -471,7 +475,7 @@ def list_albums(params): | |||||||
|         link_next = navigate_next(params) |         link_next = navigate_next(params) | ||||||
|         listing.append(link_next) |         listing.append(link_next) | ||||||
|  |  | ||||||
|     return plugin.create_listing( |     add_directory_items(create_listing( | ||||||
|         listing, |         listing, | ||||||
|         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. |         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. | ||||||
|         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  |         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  | ||||||
| @ -479,7 +483,7 @@ def list_albums(params): | |||||||
|         sort_methods = get_sort_methods('albums',params),  |         sort_methods = get_sort_methods('albums',params),  | ||||||
|         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). |         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). | ||||||
|         content = 'albums' #string - current plugin content, e.g. ‘movies’ or ‘episodes’. |         content = 'albums' #string - current plugin content, e.g. ‘movies’ or ‘episodes’. | ||||||
|     ) |     )) | ||||||
|  |  | ||||||
| @plugin.action() | @plugin.action() | ||||||
| #@plugin.cached(cachetime) #cache (in minutes) | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
| @ -513,16 +517,16 @@ def list_tracks(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     # Album |     # Album | ||||||
|     if 'album_id' in params: |     if 'album_id' in params: | ||||||
|         generator = connection.walk_album(params['album_id']) |         generator = walk_album(params['album_id']) | ||||||
|          |          | ||||||
|     # Playlist |     # Playlist | ||||||
|     elif 'playlist_id' in params: |     elif 'playlist_id' in params: | ||||||
|         generator = connection.walk_playlist(params['playlist_id']) |         generator = walk_playlist(params['playlist_id']) | ||||||
|          |          | ||||||
|         #TO FIX |         #TO FIX | ||||||
|         #tracknumber = 0 |         #tracknumber = 0 | ||||||
| @ -532,12 +536,12 @@ def list_tracks(params): | |||||||
|          |          | ||||||
|     # Starred |     # Starred | ||||||
|     elif menu_id == 'tracks_starred': |     elif menu_id == 'tracks_starred': | ||||||
|         generator = connection.walk_tracks_starred() |         generator = walk_tracks_starred() | ||||||
|          |          | ||||||
|      |      | ||||||
|     # Random |     # Random | ||||||
|     elif menu_id == 'tracks_random': |     elif menu_id == 'tracks_random': | ||||||
|         generator = connection.walk_tracks_random(**query_args) |         generator = walk_tracks_random(**query_args) | ||||||
|     # Filters |     # Filters | ||||||
|     #else: |     #else: | ||||||
|         #TO WORK |         #TO WORK | ||||||
| @ -545,7 +549,7 @@ def list_tracks(params): | |||||||
|     #make a list out of the generator so we can iterate it several times |     #make a list out of the generator so we can iterate it several times | ||||||
|     items = list(generator) |     items = list(generator) | ||||||
|  |  | ||||||
|     #check if there is only one artist for this album (and then hide it) |     #check if there==only one artist for this album (and then hide it) | ||||||
|     artists = [item.get('artist',None) for item in items] |     artists = [item.get('artist',None) for item in items] | ||||||
|     if len(artists) <= 1: |     if len(artists) <= 1: | ||||||
|         params['hide_artist']   = True |         params['hide_artist']   = True | ||||||
| @ -573,7 +577,7 @@ def list_tracks(params): | |||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     return plugin.create_listing( |     add_directory_items(create_listing( | ||||||
|         listing, |         listing, | ||||||
|         #succeeded =        True, #if False Kodi won’t open a new listing and stays on the current level. |         #succeeded =        True, #if False Kodi won’t open a new listing and stays on the current level. | ||||||
|         #update_listing =   False, #if True, Kodi won’t open a sub-listing but refresh the current one.  |         #update_listing =   False, #if True, Kodi won’t open a sub-listing but refresh the current one.  | ||||||
| @ -581,9 +585,9 @@ def list_tracks(params): | |||||||
|         sort_methods=       get_sort_methods('tracks',params), |         sort_methods=       get_sort_methods('tracks',params), | ||||||
|         #view_mode =        None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). |         #view_mode =        None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). | ||||||
|         content =           'songs' #string - current plugin content, e.g. ‘movies’ or ‘episodes’. |         content =           'songs' #string - current plugin content, e.g. ‘movies’ or ‘episodes’. | ||||||
|     ) |     )) | ||||||
|  |  | ||||||
| #stars (persistent) cache is used to know what context action (star/unstar) we should display. | #stars (persistent) cache==used to know what context action (star/unstar) we should display. | ||||||
| #run this function every time we get starred items. | #run this function every time we get starred items. | ||||||
| #ids can be a single ID or a list | #ids can be a single ID or a list | ||||||
| #using a set makes sure that IDs will be unique. | #using a set makes sure that IDs will be unique. | ||||||
| @ -594,13 +598,13 @@ def list_playlists(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     listing = [] |     listing = [] | ||||||
|  |  | ||||||
|     # Get items |     # Get items | ||||||
|     items = connection.walk_playlists() |     items = walk_playlists() | ||||||
|  |  | ||||||
|     # Iterate through items |     # Iterate through items | ||||||
|  |  | ||||||
| @ -608,7 +612,7 @@ def list_playlists(params): | |||||||
|         entry = get_entry_playlist(item,params) |         entry = get_entry_playlist(item,params) | ||||||
|         listing.append(entry) |         listing.append(entry) | ||||||
|          |          | ||||||
|     return plugin.create_listing( |     add_directory_items(create_listing( | ||||||
|         listing, |         listing, | ||||||
|         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. |         #succeeded = True, #if False Kodi won’t open a new listing and stays on the current level. | ||||||
|         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  |         #update_listing = False, #if True, Kodi won’t open a sub-listing but refresh the current one.  | ||||||
| @ -616,7 +620,7 @@ def list_playlists(params): | |||||||
|         sort_methods = get_sort_methods('playlists',params), #he list of integer constants representing virtual folder sort methods. |         sort_methods = get_sort_methods('playlists',params), #he list of integer constants representing virtual folder sort methods. | ||||||
|         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). |         #view_mode = None, #a numeric code for a skin view mode. View mode codes are different in different skins except for 50 (basic listing). | ||||||
|         #content = None #string - current plugin content, e.g. ‘movies’ or ‘episodes’. |         #content = None #string - current plugin content, e.g. ‘movies’ or ‘episodes’. | ||||||
|     ) |     )) | ||||||
| @plugin.action() | @plugin.action() | ||||||
| #@plugin.cached(cachetime) #cache (in minutes) | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
| def search(params): | def search(params): | ||||||
| @ -630,7 +634,7 @@ def search(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|  |  | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     listing = [] |     listing = [] | ||||||
| @ -646,7 +650,7 @@ def search(params): | |||||||
|         plugin.log('One single Media Folder found; do return listing from browse_indexes()...') |         plugin.log('One single Media Folder found; do return listing from browse_indexes()...') | ||||||
|         return browse_indexes(params) |         return browse_indexes(params) | ||||||
|     else: |     else: | ||||||
|         return plugin.create_listing(listing) |         add_directory_items(create_listing(listing)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -659,7 +663,7 @@ def play_track(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     url = connection.streamUrl(sid=id, |     url = connection.streamUrl(sid=id, | ||||||
| @ -667,11 +671,11 @@ def play_track(params): | |||||||
|         tformat=Addon().get_setting('transcode_format_streaming') |         tformat=Addon().get_setting('transcode_format_streaming') | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     return url |     #return url | ||||||
|  |     _set_resolved_url(resolve_url(url)) | ||||||
|  |  | ||||||
| @plugin.action() | @plugin.action() | ||||||
| def star_item(params): | def star_item(params): | ||||||
|  |  | ||||||
|     ids=     params.get('ids'); #can be single or lists of IDs |     ids=     params.get('ids'); #can be single or lists of IDs | ||||||
|     unstar=  params.get('unstar',False); |     unstar=  params.get('unstar',False); | ||||||
|     unstar = (unstar) and (unstar != 'None') and (unstar != 'False') #TO FIX better statement ? |     unstar = (unstar) and (unstar != 'None') and (unstar != 'False') #TO FIX better statement ? | ||||||
| @ -697,7 +701,7 @@ def star_item(params): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     ### |     ### | ||||||
| @ -709,15 +713,12 @@ def star_item(params): | |||||||
|             request = connection.unstar(sids, albumIds, artistIds) |             request = connection.unstar(sids, albumIds, artistIds) | ||||||
|         else: |         else: | ||||||
|             request = connection.star(sids, albumIds, artistIds) |             request = connection.star(sids, albumIds, artistIds) | ||||||
|  |  | ||||||
|         if request['status'] == 'ok': |         if request['status'] == 'ok': | ||||||
|             did_action = True |             did_action = True | ||||||
|  |  | ||||||
|     except: |     except: | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     ### |  | ||||||
|      |  | ||||||
|     if did_action: |     if did_action: | ||||||
|          |          | ||||||
|         if unstar: |         if unstar: | ||||||
| @ -740,8 +741,8 @@ def star_item(params): | |||||||
|         else: |         else: | ||||||
|             plugin.log_error('Unable to star %s #%s' % (type,json.dumps(ids))) |             plugin.log_error('Unable to star %s #%s' % (type,json.dumps(ids))) | ||||||
|  |  | ||||||
|     return did_action |     #return did_action | ||||||
|          |     return     | ||||||
|  |  | ||||||
|          |          | ||||||
| @plugin.action() | @plugin.action() | ||||||
| @ -775,6 +776,7 @@ def download_item(params): | |||||||
|  |  | ||||||
|     return did_action |     return did_action | ||||||
|  |  | ||||||
|  | #@plugin.cached(cachetime) #cache (in minutes)     | ||||||
| def get_entry_playlist(item,params): | def get_entry_playlist(item,params): | ||||||
|     image = connection.getCoverArtUrl(item.get('coverArt')) |     image = connection.getCoverArtUrl(item.get('coverArt')) | ||||||
|     return { |     return { | ||||||
| @ -796,6 +798,7 @@ def get_entry_playlist(item,params): | |||||||
|     } |     } | ||||||
|  |  | ||||||
| #star (or unstar) an item | #star (or unstar) an item | ||||||
|  | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
| def get_entry_artist(item,params): | def get_entry_artist(item,params): | ||||||
|     image = connection.getCoverArtUrl(item.get('coverArt')) |     image = connection.getCoverArtUrl(item.get('coverArt')) | ||||||
|     return { |     return { | ||||||
| @ -815,6 +818,7 @@ def get_entry_artist(item,params): | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
| def get_entry_album(item, params): | def get_entry_album(item, params): | ||||||
|      |      | ||||||
|     image = connection.getCoverArtUrl(item.get('coverArt')) |     image = connection.getCoverArtUrl(item.get('coverArt')) | ||||||
| @ -857,6 +861,7 @@ def get_entry_album(item, params): | |||||||
|  |  | ||||||
|     return entry |     return entry | ||||||
|  |  | ||||||
|  | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
| def get_entry_track(item,params): | def get_entry_track(item,params): | ||||||
|      |      | ||||||
|     menu_id = params.get('menu_id') |     menu_id = params.get('menu_id') | ||||||
| @ -904,11 +909,21 @@ def get_entry_track(item,params): | |||||||
|  |  | ||||||
|     return entry |     return entry | ||||||
|  |  | ||||||
|  | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
| def get_starred_label(id,label): | def get_starred_label(id,label): | ||||||
|     if is_starred(id): |     if is_starred(id): | ||||||
|         label = '[COLOR=FF00FF00]%s[/COLOR]' % label |         label = '[COLOR=FF00FF00]%s[/COLOR]' % label | ||||||
|     return label |     return label | ||||||
|  |  | ||||||
|  | def is_starred(id): | ||||||
|  |     starred = stars_cache_get() | ||||||
|  |     id = int(id) | ||||||
|  |     if id in starred: | ||||||
|  |         return True | ||||||
|  |     else: | ||||||
|  |         return False | ||||||
|  |  | ||||||
|  | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
| def get_entry_track_label(item,hide_artist = False): | def get_entry_track_label(item,hide_artist = False): | ||||||
|     if hide_artist: |     if hide_artist: | ||||||
|         label = item.get('title', '<Unknown>') |         label = item.get('title', '<Unknown>') | ||||||
| @ -920,6 +935,7 @@ def get_entry_track_label(item,hide_artist = False): | |||||||
|  |  | ||||||
|     return get_starred_label(item.get('id'),label) |     return get_starred_label(item.get('id'),label) | ||||||
|  |  | ||||||
|  | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
| def get_entry_album_label(item,hide_artist = False): | def get_entry_album_label(item,hide_artist = False): | ||||||
|     if hide_artist: |     if hide_artist: | ||||||
|         label = item.get('name', '<Unknown>') |         label = item.get('name', '<Unknown>') | ||||||
| @ -928,7 +944,7 @@ def get_entry_album_label(item,hide_artist = False): | |||||||
|                              item.get('name', '<Unknown>')) |                              item.get('name', '<Unknown>')) | ||||||
|     return get_starred_label(item.get('id'),label) |     return get_starred_label(item.get('id'),label) | ||||||
|  |  | ||||||
|  | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
| def get_sort_methods(type,params): | def get_sort_methods(type,params): | ||||||
|     #sort method for list types |     #sort method for list types | ||||||
|     #https://github.com/xbmc/xbmc/blob/master/xbmc/SortFileItem.h |     #https://github.com/xbmc/xbmc/blob/master/xbmc/SortFileItem.h | ||||||
| @ -947,7 +963,7 @@ def get_sort_methods(type,params): | |||||||
|         xbmcplugin.SORT_METHOD_UNSORTED |         xbmcplugin.SORT_METHOD_UNSORTED | ||||||
|     ] |     ] | ||||||
|      |      | ||||||
|     if type is 'artists': |     if type=='artists': | ||||||
|          |          | ||||||
|         artists = [ |         artists = [ | ||||||
|             xbmcplugin.SORT_METHOD_ARTIST |             xbmcplugin.SORT_METHOD_ARTIST | ||||||
| @ -955,7 +971,7 @@ def get_sort_methods(type,params): | |||||||
|  |  | ||||||
|         sortable = sortable + artists |         sortable = sortable + artists | ||||||
|          |          | ||||||
|     elif type is 'albums': |     elif type=='albums': | ||||||
|          |          | ||||||
|         albums = [ |         albums = [ | ||||||
|             xbmcplugin.SORT_METHOD_ALBUM, |             xbmcplugin.SORT_METHOD_ALBUM, | ||||||
| @ -969,7 +985,7 @@ def get_sort_methods(type,params): | |||||||
|  |  | ||||||
|         sortable = sortable + albums |         sortable = sortable + albums | ||||||
|          |          | ||||||
|     elif type is 'tracks': |     elif type=='tracks': | ||||||
|  |  | ||||||
|         tracks = [ |         tracks = [ | ||||||
|             xbmcplugin.SORT_METHOD_TITLE, |             xbmcplugin.SORT_METHOD_TITLE, | ||||||
| @ -992,7 +1008,7 @@ def get_sort_methods(type,params): | |||||||
|          |          | ||||||
|         sortable = sortable + tracks |         sortable = sortable + tracks | ||||||
|          |          | ||||||
|     elif type is 'playlists': |     elif type=='playlists': | ||||||
|  |  | ||||||
|         playlists = [ |         playlists = [ | ||||||
|             xbmcplugin.SORT_METHOD_TITLE, |             xbmcplugin.SORT_METHOD_TITLE, | ||||||
| @ -1010,7 +1026,7 @@ def stars_cache_update(ids,remove=False): | |||||||
|     #get existing cache set |     #get existing cache set | ||||||
|     starred = stars_cache_get() |     starred = stars_cache_get() | ||||||
|      |      | ||||||
|     #make sure this is a list |     #make sure this==a list | ||||||
|     if not isinstance(ids, list): |     if not isinstance(ids, list): | ||||||
|         ids = [ids] |         ids = [ids] | ||||||
|      |      | ||||||
| @ -1034,21 +1050,19 @@ def stars_cache_update(ids,remove=False): | |||||||
|     plugin.log(starred) |     plugin.log(starred) | ||||||
|  |  | ||||||
|  |  | ||||||
| def stars_cache_get(): | def stars_cache_get(): #Retrieving stars from cache is too slow, so load to local variable | ||||||
|     with plugin.get_storage() as storage: |     global local_starred     | ||||||
|         starred = storage.get('starred_ids',set()) |     plugin.log(len(local_starred))    | ||||||
|  |     if(len(local_starred)>0): | ||||||
|     plugin.log('stars_cache_get:') |         plugin.log('stars already loaded:') | ||||||
|     plugin.log(starred) |         plugin.log(local_starred)         | ||||||
|     return starred |         return(local_starred) | ||||||
|  |  | ||||||
| def is_starred(id): |  | ||||||
|     starred = stars_cache_get() |  | ||||||
|     id = int(id) |  | ||||||
|     if id in starred: |  | ||||||
|         return True |  | ||||||
|     else: |     else: | ||||||
|         return False |         with plugin.get_storage() as storage: | ||||||
|  |             local_starred = storage.get('starred_ids',set()) | ||||||
|  |             plugin.log('stars_cache_get:') | ||||||
|  |             plugin.log(local_starred) | ||||||
|  |             return local_starred | ||||||
|  |  | ||||||
| def navigate_next(params): | def navigate_next(params): | ||||||
|    |    | ||||||
| @ -1093,12 +1107,13 @@ def context_action_star(type,id): | |||||||
|  |  | ||||||
|         label = Addon().get_localized_string(30034) |         label = Addon().get_localized_string(30034) | ||||||
|      |      | ||||||
|  |     xbmc.log('Context action star returning RunPlugin(%s)' % plugin.get_url(action='star_item',type=type,ids=id,unstar=starred),xbmc.LOGDEBUG) | ||||||
|     return ( |     return ( | ||||||
|         label,  |         label,  | ||||||
|         'XBMC.RunPlugin(%s)' % plugin.get_url(action='star_item',type=type,ids=id,unstar=starred) |         'RunPlugin(%s)' % plugin.get_url(action='star_item',type=type,ids=id,unstar=starred) | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
| #Subsonic API says this is supported for artist,tracks and albums, | #Subsonic API says this==supported for artist,tracks and albums, | ||||||
| #But I can see it available only for tracks on Subsonic 5.3, so disable it. | #But I can see it available only for tracks on Subsonic 5.3, so disable it. | ||||||
| def can_star(type,ids = None): | def can_star(type,ids = None): | ||||||
|      |      | ||||||
| @ -1124,11 +1139,11 @@ def context_action_download(type,id): | |||||||
|      |      | ||||||
|     return ( |     return ( | ||||||
|         label,  |         label,  | ||||||
|         'XBMC.RunPlugin(%s)' % plugin.get_url(action='download_item',type=type,id=id) |         'RunPlugin(%s)' % plugin.get_url(action='download_item',type=type,id=id) | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
| def can_download(type,id = None): | def can_download(type,id = None): | ||||||
|     if id is None: |     if id==None: | ||||||
|         return False |         return False | ||||||
|      |      | ||||||
|     if type == 'track': |     if type == 'track': | ||||||
| @ -1138,7 +1153,7 @@ def can_download(type,id = None): | |||||||
|      |      | ||||||
| def download_tracks(ids): | def download_tracks(ids): | ||||||
|  |  | ||||||
|     #popup is fired before, in download_item |     #popup==fired before, in download_item | ||||||
|     download_folder = Addon().get_setting('download_folder') |     download_folder = Addon().get_setting('download_folder') | ||||||
|     if not download_folder: |     if not download_folder: | ||||||
|         return |         return | ||||||
| @ -1163,7 +1178,7 @@ def download_tracks(ids): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     #progress... |     #progress... | ||||||
| @ -1236,7 +1251,7 @@ def download_album(id): | |||||||
|     # get connection |     # get connection | ||||||
|     connection = get_connection() |     connection = get_connection() | ||||||
|      |      | ||||||
|     if connection is False: |     if connection==False: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     # get album infos |     # get album infos | ||||||
| @ -1255,6 +1270,275 @@ def download_album(id): | |||||||
|  |  | ||||||
|     download_tracks(ids) |     download_tracks(ids) | ||||||
|  |  | ||||||
|  | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
|  | def create_listing(listing, succeeded=True, update_listing=False, cache_to_disk=False, sort_methods=None,view_mode=None, content=None, category=None): | ||||||
|  |         return ListContext(listing, succeeded, update_listing, cache_to_disk,sort_methods, view_mode, content, category) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def resolve_url(path='', play_item=None, succeeded=True): | ||||||
|  |     """ | ||||||
|  |     Create and return a context dict to resolve a playable URL | ||||||
|  |     :param path: the path to a playable media. | ||||||
|  |     :type path: str or unicode | ||||||
|  |     :param play_item: a dict of item properties as described in the class docstring. | ||||||
|  |         It allows to set additional properties for the item being played, like graphics, metadata etc. | ||||||
|  |         if ``play_item`` parameter==present, then ``path`` value==ignored, and the path must be set via | ||||||
|  |         ``'path'`` property of a ``play_item`` dict. | ||||||
|  |     :type play_item: dict | ||||||
|  |     :param succeeded: if ``False``, Kodi won't play anything | ||||||
|  |     :type succeeded: bool | ||||||
|  |     :return: context object containing necessary parameters | ||||||
|  |         for Kodi to play the selected media. | ||||||
|  |     :rtype: PlayContext | ||||||
|  |     """ | ||||||
|  |     return PlayContext(path, play_item, succeeded) | ||||||
|  |  | ||||||
|  | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
|  | def create_list_item(item): | ||||||
|  |     """ | ||||||
|  |     Create an :class:`xbmcgui.ListItem` instance from an item dict | ||||||
|  |     :param item: a dict of ListItem properties | ||||||
|  |     :type item: dict | ||||||
|  |     :return: ListItem instance | ||||||
|  |     :rtype: xbmcgui.ListItem | ||||||
|  |     """ | ||||||
|  |     major_version = xbmc.getInfoLabel('System.BuildVersion')[:2] | ||||||
|  |     if major_version >= '18': | ||||||
|  |         list_item = xbmcgui.ListItem(label=item.get('label', ''), | ||||||
|  |                                      label2=item.get('label2', ''), | ||||||
|  |                                      path=item.get('path', ''), | ||||||
|  |                                      offscreen=item.get('offscreen', False)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     art = item.get('art', {}) | ||||||
|  |     art['thumb'] = item.get('thumb', '') | ||||||
|  |     art['icon'] = item.get('icon', '') | ||||||
|  |     art['fanart'] = item.get('fanart', '') | ||||||
|  |     item['art'] = art | ||||||
|  |     cont_look = item.get('content_lookup') | ||||||
|  |     if cont_look is not None: | ||||||
|  |         list_item.setContentLookup(cont_look) | ||||||
|  |     if item.get('art'): | ||||||
|  |         list_item.setArt(item['art']) | ||||||
|  |     if item.get('stream_info'): | ||||||
|  |         for stream, stream_info in item['stream_info'].items(): | ||||||
|  |             list_item.addStreamInfo(stream, stream_info) | ||||||
|  |     if item.get('info'): | ||||||
|  |         for media, info in item['info'].items(): | ||||||
|  |             list_item.setInfo(media, info) | ||||||
|  |     if item.get('context_menu') is not None: | ||||||
|  |         list_item.addContextMenuItems(item['context_menu']) | ||||||
|  |     if item.get('subtitles'): | ||||||
|  |         list_item.setSubtitles(item['subtitles']) | ||||||
|  |     if item.get('mime'): | ||||||
|  |         list_item.setMimeType(item['mime']) | ||||||
|  |     if item.get('properties'): | ||||||
|  |         for key, value in item['properties'].items(): | ||||||
|  |             list_item.setProperty(key, value) | ||||||
|  |     if major_version >= '17': | ||||||
|  |         cast = item.get('cast') | ||||||
|  |         if cast is not None: | ||||||
|  |             list_item.setCast(cast) | ||||||
|  |         db_ids = item.get('online_db_ids') | ||||||
|  |         if db_ids is not None: | ||||||
|  |             list_item.setUniqueIDs(db_ids) | ||||||
|  |         ratings = item.get('ratings') | ||||||
|  |         if ratings is not None: | ||||||
|  |             for rating in ratings: | ||||||
|  |                 list_item.setRating(**rating) | ||||||
|  |     return list_item | ||||||
|  |  | ||||||
|  | def _set_resolved_url(context): | ||||||
|  |  | ||||||
|  |     plugin.log_debug('Resolving URL from {0}'.format(str(context))) | ||||||
|  |     if context.play_item==None: | ||||||
|  |         list_item = xbmcgui.ListItem(path=context.path) | ||||||
|  |     else: | ||||||
|  |         list_item = self.create_list_item(context.play_item) | ||||||
|  |     xbmcplugin.setResolvedUrl(plugin.handle, context.succeeded, list_item) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #@plugin.cached(cachetime) #cache (in minutes) | ||||||
|  | def add_directory_items(context): | ||||||
|  |     plugin.log_debug('Creating listing from {0}'.format(str(context))) | ||||||
|  |     if context.category is not None: | ||||||
|  |         xbmcplugin.setPluginCategory(plugin.handle, context.category) | ||||||
|  |     if context.content is not None: | ||||||
|  |         xbmcplugin.setContent(plugin.handle, context.content)  # This must be at the beginning | ||||||
|  |     for item in context.listing: | ||||||
|  |         is_folder = item.get('is_folder', True) | ||||||
|  |         if item.get('list_item') is not None: | ||||||
|  |             list_item = item['list_item'] | ||||||
|  |         else: | ||||||
|  |             list_item = create_list_item(item) | ||||||
|  |             if item.get('is_playable'): | ||||||
|  |                 list_item.setProperty('IsPlayable', 'true') | ||||||
|  |                 is_folder = False | ||||||
|  |         xbmcplugin.addDirectoryItem(plugin.handle, item['url'], list_item, is_folder) | ||||||
|  |     if context.sort_methods is not None: | ||||||
|  |         if isinstance(context.sort_methods, (int, dict)): | ||||||
|  |             sort_methods = [context.sort_methods] | ||||||
|  |         elif isinstance(context.sort_methods, (tuple, list)): | ||||||
|  |             sort_methods = context.sort_methods | ||||||
|  |         else: | ||||||
|  |             raise TypeError( | ||||||
|  |                 'sort_methods parameter must be of int, dict, tuple or list type!') | ||||||
|  |         for method in sort_methods: | ||||||
|  |             if isinstance(method, int): | ||||||
|  |                 xbmcplugin.addSortMethod(plugin.handle, method) | ||||||
|  |             elif isinstance(method, dict): | ||||||
|  |                 xbmcplugin.addSortMethod(plugin.handle, **method) | ||||||
|  |             else: | ||||||
|  |                 raise TypeError( | ||||||
|  |                     'method parameter must be of int or dict type!') | ||||||
|  |  | ||||||
|  |     xbmcplugin.endOfDirectory(plugin.handle, | ||||||
|  |                               context.succeeded, | ||||||
|  |                               context.update_listing, | ||||||
|  |                               context.cache_to_disk) | ||||||
|  |     if context.view_mode is not None: | ||||||
|  |         xbmc.executebuiltin('Container.SetViewMode({0})'.format(context.view_mode)) | ||||||
|  |  | ||||||
|  | def walk_index(folder_id=None): | ||||||
|  |     """ | ||||||
|  |     Request Subsonic's index and iterate each item. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     response = connection.getIndexes(folder_id) | ||||||
|  |  | ||||||
|  |     for index in response["indexes"]["index"]: | ||||||
|  |         for artist in index["artist"]: | ||||||
|  |             yield artist | ||||||
|  |              | ||||||
|  | def walk_playlists(): | ||||||
|  |     """ | ||||||
|  |     Request Subsonic's playlists and iterate over each item. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     response = connection.getPlaylists() | ||||||
|  |  | ||||||
|  |     for child in response["playlists"]["playlist"]: | ||||||
|  |         yield child | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def walk_playlist(playlist_id): | ||||||
|  |     """ | ||||||
|  |     Request Subsonic's playlist items and iterate over each item. | ||||||
|  |     """ | ||||||
|  |     response = connection.getPlaylist(playlist_id) | ||||||
|  |  | ||||||
|  |     for child in response["playlist"]["entry"]: | ||||||
|  |         yield child | ||||||
|  |  | ||||||
|  | def walk_folders(): | ||||||
|  |     response = connection.getMusicFolders() | ||||||
|  |      | ||||||
|  |     for child in response["musicFolders"]["musicFolder"]: | ||||||
|  |         yield child | ||||||
|  |  | ||||||
|  | def walk_directory(directory_id): | ||||||
|  |     """ | ||||||
|  |     Request a Subsonic music directory and iterate over each item. | ||||||
|  |     """ | ||||||
|  |     response = connection.getMusicDirectory(directory_id) | ||||||
|  |      | ||||||
|  |     try: | ||||||
|  |         for child in response["directory"]["child"]: | ||||||
|  |             if child.get("isDir"): | ||||||
|  |                 for child in walk_directory(child["id"]): | ||||||
|  |                     yield child | ||||||
|  |             else: | ||||||
|  |                 yield child | ||||||
|  |     except: | ||||||
|  |         yield from () | ||||||
|  |  | ||||||
|  | def walk_artist(artist_id): | ||||||
|  |     """ | ||||||
|  |     Request a Subsonic artist and iterate over each album. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     response = connection.getArtist(artist_id) | ||||||
|  |  | ||||||
|  |     for child in response["artist"]["album"]: | ||||||
|  |         yield child | ||||||
|  |  | ||||||
|  | def walk_artists(): | ||||||
|  |     """ | ||||||
|  |     (ID3 tags) | ||||||
|  |     Request all artists and iterate over each item. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     response = connection.getArtists() | ||||||
|  |  | ||||||
|  |     for index in response["artists"]["index"]: | ||||||
|  |         for artist in index["artist"]: | ||||||
|  |             yield artist | ||||||
|  |  | ||||||
|  | def walk_genres(): | ||||||
|  |     """ | ||||||
|  |     (ID3 tags) | ||||||
|  |     Request all genres and iterate over each item. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     response = connection.getGenres() | ||||||
|  |  | ||||||
|  |     for genre in response["genres"]["genre"]: | ||||||
|  |         yield genre | ||||||
|  |  | ||||||
|  | def walk_albums(ltype, size=None, fromYear=None,toYear=None, genre=None, offset=None): | ||||||
|  |     """ | ||||||
|  |     (ID3 tags) | ||||||
|  |     Request all albums for a given genre and iterate over each album. | ||||||
|  |     """ | ||||||
|  |      | ||||||
|  |     if ltype == 'byGenre' and genre is None: | ||||||
|  |         return | ||||||
|  |      | ||||||
|  |     if ltype == 'byYear' and (fromYear is None or toYear is None): | ||||||
|  |         return | ||||||
|  |  | ||||||
|  |     response = connection.getAlbumList2( | ||||||
|  |         ltype=ltype, size=size, fromYear=fromYear, toYear=toYear,genre=genre, offset=offset) | ||||||
|  |  | ||||||
|  |     if not response["albumList2"]["album"]: | ||||||
|  |         return | ||||||
|  |  | ||||||
|  |     for album in response["albumList2"]["album"]: | ||||||
|  |         yield album | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def walk_album(album_id): | ||||||
|  |     """ | ||||||
|  |     (ID3 tags) | ||||||
|  |     Request an album and iterate over each item. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     response = connection.getAlbum(album_id) | ||||||
|  |  | ||||||
|  |     for song in response["album"]["song"]: | ||||||
|  |         yield song | ||||||
|  |  | ||||||
|  | def walk_tracks_random(size=None, genre=None, fromYear=None,toYear=None): | ||||||
|  |     """ | ||||||
|  |     Request random songs by genre and/or year and iterate over each song. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     response = connection.getRandomSongs( | ||||||
|  |         size=size, genre=genre, fromYear=fromYear, toYear=toYear) | ||||||
|  |  | ||||||
|  |     for song in response["randomSongs"]["song"]: | ||||||
|  |         yield song | ||||||
|  |          | ||||||
|  |  | ||||||
|  | def walk_tracks_starred(): | ||||||
|  |     """ | ||||||
|  |     Request Subsonic's starred songs and iterate over each item. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     response = connection.getStarred() | ||||||
|  |  | ||||||
|  |     for song in response["starred"]["song"]: | ||||||
|  |         yield song | ||||||
|  |  | ||||||
|  |  | ||||||
| # Start plugin from within Kodi. | # Start plugin from within Kodi. | ||||||
| @ -1262,13 +1546,3 @@ if __name__ == "__main__": | |||||||
|     # Map actions |     # Map actions | ||||||
|     # Note that we map callable objects without brackets () |     # Note that we map callable objects without brackets () | ||||||
|     plugin.run() |     plugin.run() | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|      |  | ||||||
|      |  | ||||||
|      |  | ||||||
|      |  | ||||||
|      |  | ||||||
|  | |||||||
| @ -157,4 +157,15 @@ msgctxt "#30039" | |||||||
| msgid "Search" | msgid "Search" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
|  | msgctxt "#30040" | ||||||
|  | msgid "useGET" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | msgctxt "#30041" | ||||||
|  | msgid "legacyauth" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | msgctxt "#30042" | ||||||
|  | msgid "port" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | |||||||
| @ -152,9 +152,20 @@ msgctxt "#30038" | |||||||
| msgid "Browse" | msgid "Browse" | ||||||
| msgstr "Parcourir" | msgstr "Parcourir" | ||||||
|  |  | ||||||
|  |  | ||||||
| msgctxt "#30039" | msgctxt "#30039" | ||||||
| msgid "Search" | msgid "Search" | ||||||
| msgstr "Rechercher" | msgstr "Rechercher" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | msgctxt "#30040" | ||||||
|  | msgid "useGET" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | msgctxt "#30041" | ||||||
|  | msgid "legacyauth" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | msgctxt "#30042" | ||||||
|  | msgid "port" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | |||||||
| @ -155,3 +155,16 @@ msgstr "Durchsuchen" | |||||||
| msgctxt "#30039" | msgctxt "#30039" | ||||||
| msgid "Search" | msgid "Search" | ||||||
| msgstr "Suche" | msgstr "Suche" | ||||||
|  |  | ||||||
|  | msgctxt "#30040" | ||||||
|  | msgid "useGET" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | msgctxt "#30041" | ||||||
|  | msgid "legacyauth" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | msgctxt "#30042" | ||||||
|  | msgid "port" | ||||||
|  | msgstr "" | ||||||
|  |  | ||||||
|  | |||||||
| @ -4,6 +4,7 @@ | |||||||
|     <category label="30000"> |     <category label="30000"> | ||||||
|         <setting label="30001" type="lsep" /> |         <setting label="30001" type="lsep" /> | ||||||
|         <setting label="30002" id="subsonic_url" type="text"  default="http://demo.subsonic.org"/> |         <setting label="30002" id="subsonic_url" type="text"  default="http://demo.subsonic.org"/> | ||||||
|  |         <setting label="30042" id="port" type="text"  default="80"/> | ||||||
|         <setting label="30003" id="username" type="text" default="guest3"/> |         <setting label="30003" id="username" type="text" default="guest3"/> | ||||||
|         <setting label="30004" id="password" type="text" option="hidden"  default="guest"/> |         <setting label="30004" id="password" type="text" option="hidden"  default="guest"/> | ||||||
|         <setting label="30005" type="lsep" /> |         <setting label="30005" type="lsep" /> | ||||||
| @ -13,14 +14,16 @@ | |||||||
|         <setting label="30009" id="download_folder"  type="folder"  source="auto" option="writeable"/> |         <setting label="30009" id="download_folder"  type="folder"  source="auto" option="writeable"/> | ||||||
|         <setting label="30010" type="lsep" /> |         <setting label="30010" type="lsep" /> | ||||||
|         <setting label="30011" id="transcode_format_streaming" type="labelenum"  values="mp3|raw|flv|ogg"/> |         <setting label="30011" id="transcode_format_streaming" type="labelenum"  values="mp3|raw|flv|ogg"/> | ||||||
|         <setting label="30012" id="bitrate_streaming" type="labelenum"  values="320|256|224|192|160|128|112|96|80|64|56|48|40|32"/> |         <setting label="30012" id="bitrate_streaming" type="labelenum" default="0" values="320|256|224|192|160|128|112|96|80|64|56|48|40|32|0"/> | ||||||
|     </category> |     </category> | ||||||
|      |      | ||||||
|     <!-- ADVANCED --> |     <!-- ADVANCED --> | ||||||
|   	<category label="30013"> |   	<category label="30013"> | ||||||
|         <setting label="30001" type="lsep" /> |         <setting label="30001" type="lsep" /> | ||||||
|         <setting label="30014" id="apiversion" type="labelenum"  values="1.11.0|1.12.0|1.13.0|1.14.0" default="1.13.0"/> |         <setting label="30014" id="apiversion" type="labelenum"  values="1.11.0|1.12.0|1.13.0|1.14.0|1.15.0|1.16.0" default="1.15.0"/> | ||||||
| 		<setting label="30016" id="insecure" type="bool"  default="false" /> | 		<setting label="30016" id="insecure" type="bool"  default="false" /> | ||||||
|  | 		<setting label="30040" id="useget" type="bool"  default="false" /> | ||||||
|  | 		<setting label="30041" id="legacyauth" type="bool" default="false" /> | ||||||
|         <setting label="30017" type="lsep" /> |         <setting label="30017" type="lsep" /> | ||||||
|         <setting label="30018" id="cachetime" type="labelenum"  default="15" values="1|5|15|30|60|120|180|720|1440"/> |         <setting label="30018" id="cachetime" type="labelenum"  default="15" values="1|5|15|30|60|120|180|720|1440"/> | ||||||
| 		 | 		 | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user