mirror of
https://github.com/gordielachance/plugin.audio.subsonic
synced 2025-01-09 14:44:32 +01:00
8a786b1eae
was crashing on KODI OpenELEC, safer not to use it.
2771 lines
102 KiB
Python
2771 lines
102 KiB
Python
"""
|
|
This file is part of py-sonic.
|
|
|
|
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
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
py-sonic is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with py-sonic. If not, see <http://www.gnu.org/licenses/>
|
|
"""
|
|
|
|
from urllib import urlencode
|
|
from .errors import *
|
|
from pprint import pprint
|
|
from cStringIO import StringIO
|
|
from netrc import netrc
|
|
from hashlib import md5
|
|
import json, urllib2, httplib, logging, socket, ssl, sys, os
|
|
|
|
API_VERSION = '1.14.0'
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
class HTTPSConnectionChain(httplib.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):
|
|
sock = socket.create_connection((self.host, self.port), self.timeout)
|
|
if self._tunnel_host:
|
|
self.sock = sock
|
|
self._tunnel()
|
|
return sock
|
|
|
|
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()
|
|
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:
|
|
self.sock = ssl.wrap_socket(sock, self.key_file, self.cert_file,
|
|
ssl_version=proto)
|
|
except:
|
|
sock.close()
|
|
else:
|
|
# Cache the working ssl version
|
|
HTTPSConnectionChain._ssl_working_proto = proto
|
|
break
|
|
|
|
|
|
class HTTPSHandlerChain(urllib2.HTTPSHandler):
|
|
def https_open(self, req):
|
|
return self.do_open(HTTPSConnectionChain, req)
|
|
|
|
# install opener
|
|
urllib2.install_opener(urllib2.build_opener(HTTPSHandlerChain()))
|
|
|
|
class PysHTTPRedirectHandler(urllib2.HTTPRedirectHandler):
|
|
"""
|
|
This class is used to override the default behavior of the
|
|
HTTPRedirectHandler, which does *not* redirect POST data
|
|
"""
|
|
def redirect_request(self, req, fp, code, msg, headers, newurl):
|
|
m = req.get_method()
|
|
if (code in (301, 302, 303, 307) and m in ("GET", "HEAD")
|
|
or code in (301, 302, 303) and m == "POST"):
|
|
newurl = newurl.replace(' ', '%20')
|
|
newheaders = dict((k, v) for k, v in req.headers.items()
|
|
if k.lower() not in ("content-length", "content-type")
|
|
)
|
|
data = None
|
|
if req.has_data():
|
|
data = req.get_data()
|
|
return urllib2.Request(newurl,
|
|
data=data,
|
|
headers=newheaders,
|
|
origin_req_host=req.get_origin_req_host(),
|
|
unverifiable=True)
|
|
else:
|
|
raise urllib2.HTTPError(req.get_full_url(), code, msg, headers, fp)
|
|
|
|
class Connection(object):
|
|
def __init__(self, baseUrl, username=None, password=None, port=4040,
|
|
serverPath='/rest', appName='py-sonic', apiVersion=API_VERSION,
|
|
insecure=False, useNetrc=None, legacyAuth=False, useGET=False):
|
|
"""
|
|
This will create a connection to your subsonic server
|
|
|
|
baseUrl:str The base url for your server. Be sure to use
|
|
"https" for SSL connections. If you are using
|
|
a port other than the default 4040, be sure to
|
|
specify that with the port argument. Do *not*
|
|
append it here.
|
|
|
|
ex: http://subsonic.example.com
|
|
|
|
If you are running subsonic under a different
|
|
path, specify that with the "serverPath" arg,
|
|
*not* here. For example, if your subsonic
|
|
lives at:
|
|
|
|
https://mydomain.com:8080/path/to/subsonic/rest
|
|
|
|
You would set the following:
|
|
|
|
baseUrl = "https://mydomain.com"
|
|
port = 8080
|
|
serverPath = "/path/to/subsonic/rest"
|
|
username:str The username to use for the connection. This
|
|
can be None if `useNetrc' is True (and you
|
|
have a valid entry in your netrc file)
|
|
password:str The password to use for the connection. This
|
|
can be None if `useNetrc' is True (and you
|
|
have a valid entry in your netrc file)
|
|
port:int The port number to connect on. The default for
|
|
unencrypted subsonic connections is 4040
|
|
serverPath:str The base resource path for the subsonic views.
|
|
This is useful if you have your subsonic server
|
|
behind a proxy and the path that you are proxying
|
|
is different from the default of '/rest'.
|
|
Ex:
|
|
serverPath='/path/to/subs'
|
|
|
|
The full url that would be built then would be
|
|
(assuming defaults and using "example.com" and
|
|
you are using the "ping" view):
|
|
|
|
http://example.com:4040/path/to/subs/ping.view
|
|
appName:str The name of your application.
|
|
apiVersion:str The API version you wish to use for your
|
|
application. Subsonic will throw an error if you
|
|
try to use/send an api version higher than what
|
|
the server supports. See the Subsonic API docs
|
|
to find the Subsonic version -> API version table.
|
|
This is useful if you are connecting to an older
|
|
version of Subsonic.
|
|
insecure:bool This will allow you to use self signed
|
|
certificates when connecting if set to True.
|
|
useNetrc:str|bool You can either specify a specific netrc
|
|
formatted file or True to use your default
|
|
netrc file ($HOME/.netrc).
|
|
legacyAuth:bool Use pre-1.13.0 API version authentication
|
|
useGET:bool Use a GET request instead of the default POST
|
|
request. This is not recommended as request
|
|
URLs can get very long with some API calls
|
|
"""
|
|
self._baseUrl = baseUrl
|
|
self._hostname = baseUrl.split('://')[1].strip()
|
|
self._username = username
|
|
self._rawPass = password
|
|
self._legacyAuth = legacyAuth
|
|
self._useGET = useGET
|
|
|
|
self._netrc = None
|
|
if useNetrc is not None:
|
|
self._process_netrc(useNetrc)
|
|
elif username is None or password is None:
|
|
raise CredentialError('You must specify either a username/password '
|
|
'combination or "useNetrc" must be either True or a string '
|
|
'representing a path to a netrc file')
|
|
|
|
self._port = int(port)
|
|
self._apiVersion = apiVersion
|
|
self._appName = appName
|
|
self._serverPath = serverPath.strip('/')
|
|
self._insecure = insecure
|
|
self._opener = self._getOpener(self._username, self._rawPass)
|
|
|
|
# Properties
|
|
def setBaseUrl(self, url):
|
|
self._baseUrl = url
|
|
self._opener = self._getOpener(self._username, self._rawPass)
|
|
baseUrl = property(lambda s: s._baseUrl, setBaseUrl)
|
|
|
|
def setPort(self, port):
|
|
self._port = int(port)
|
|
port = property(lambda s: s._port, setPort)
|
|
|
|
def setUsername(self, username):
|
|
self._username = username
|
|
self._opener = self._getOpener(self._username, self._rawPass)
|
|
username = property(lambda s: s._username, setUsername)
|
|
|
|
def setPassword(self, password):
|
|
self._rawPass = password
|
|
# Redo the opener with the new creds
|
|
self._opener = self._getOpener(self._username, self._rawPass)
|
|
password = property(lambda s: s._rawPass, setPassword)
|
|
|
|
apiVersion = property(lambda s: s._apiVersion)
|
|
|
|
def setAppName(self, appName):
|
|
self._appName = appName
|
|
appName = property(lambda s: s._appName, setAppName)
|
|
|
|
def setServerPath(self, path):
|
|
self._serverPath = path.strip('/')
|
|
serverPath = property(lambda s: s._serverPath, setServerPath)
|
|
|
|
def setInsecure(self, insecure):
|
|
self._insecure = insecure
|
|
insecure = property(lambda s: s._insecure, setInsecure)
|
|
|
|
def setLegacyAuth(self, lauth):
|
|
self._legacyAuth = lauth
|
|
legacyAuth = property(lambda s: s._legacyAuth, setLegacyAuth)
|
|
|
|
def setGET(self, get):
|
|
self._useGET = get
|
|
useGET = property(lambda s: s._useGET, setGET)
|
|
|
|
# API methods
|
|
def ping(self):
|
|
"""
|
|
since: 1.0.0
|
|
|
|
Returns a boolean True if the server is alive, False otherwise
|
|
"""
|
|
methodName = 'ping'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
try:
|
|
res = self._doInfoReq(req)
|
|
except:
|
|
return False
|
|
if res['status'] == 'ok':
|
|
return True
|
|
elif res['status'] == 'failed':
|
|
exc = getExcByCode(res['error']['code'])
|
|
raise exc(res['error']['message'])
|
|
return False
|
|
|
|
def getLicense(self):
|
|
"""
|
|
since: 1.0.0
|
|
|
|
Gets details related to the software license
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'license': {u'date': u'2010-05-21T11:14:39',
|
|
u'email': u'email@example.com',
|
|
u'key': u'12345678901234567890123456789012',
|
|
u'valid': True},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getLicense'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getMusicFolders(self):
|
|
"""
|
|
since: 1.0.0
|
|
|
|
Returns all configured music folders
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'musicFolders': {u'musicFolder': [{u'id': 0, u'name': u'folder1'},
|
|
{u'id': 1, u'name': u'folder2'},
|
|
{u'id': 2, u'name': u'folder3'}]},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getMusicFolders'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getNowPlaying(self):
|
|
"""
|
|
since: 1.0.0
|
|
|
|
Returns what is currently being played by all users
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'nowPlaying': {u'entry': {u'album': u"Jazz 'Round Midnight 12",
|
|
u'artist': u'Astrud Gilberto',
|
|
u'bitRate': 172,
|
|
u'contentType': u'audio/mpeg',
|
|
u'coverArt': u'98349284',
|
|
u'duration': 325,
|
|
u'genre': u'Jazz',
|
|
u'id': u'2424324',
|
|
u'isDir': False,
|
|
u'isVideo': False,
|
|
u'minutesAgo': 0,
|
|
u'parent': u'542352',
|
|
u'path': u"Astrud Gilberto/Jazz 'Round Midnight 12/01 - The Girl From Ipanema.mp3",
|
|
u'playerId': 1,
|
|
u'size': 7004089,
|
|
u'suffix': u'mp3',
|
|
u'title': u'The Girl From Ipanema',
|
|
u'track': 1,
|
|
u'username': u'user1',
|
|
u'year': 1996}},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getNowPlaying'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getIndexes(self, musicFolderId=None, ifModifiedSince=0):
|
|
"""
|
|
since: 1.0.0
|
|
|
|
Returns an indexed structure of all artists
|
|
|
|
musicFolderId:int If this is specified, it will only return
|
|
artists for the given folder ID from
|
|
the getMusicFolders call
|
|
ifModifiedSince:int If specified, return a result if the artist
|
|
collection has changed since the given
|
|
unix timestamp
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'indexes': {u'index': [{u'artist': [{u'id': u'29834728934',
|
|
u'name': u'A Perfect Circle'},
|
|
{u'id': u'238472893',
|
|
u'name': u'A Small Good Thing'},
|
|
{u'id': u'9327842983',
|
|
u'name': u'A Tribe Called Quest'},
|
|
{u'id': u'29348729874',
|
|
u'name': u'A-Teens, The'},
|
|
{u'id': u'298472938',
|
|
u'name': u'ABA STRUCTURE'}],
|
|
u'lastModified': 1303318347000L},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getIndexes'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'musicFolderId': musicFolderId,
|
|
'ifModifiedSince': self._ts2milli(ifModifiedSince)})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
self._fixLastModified(res)
|
|
return res
|
|
|
|
def getMusicDirectory(self, mid):
|
|
"""
|
|
since: 1.0.0
|
|
|
|
Returns a listing of all files in a music directory. Typically used
|
|
to get a list of albums for an artist or list of songs for an album.
|
|
|
|
mid:str The string ID value which uniquely identifies the
|
|
folder. Obtained via calls to getIndexes or
|
|
getMusicDirectory. REQUIRED
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'directory': {u'child': [{u'artist': u'A Tribe Called Quest',
|
|
u'coverArt': u'223484',
|
|
u'id': u'329084',
|
|
u'isDir': True,
|
|
u'parent': u'234823940',
|
|
u'title': u'Beats, Rhymes And Life'},
|
|
{u'artist': u'A Tribe Called Quest',
|
|
u'coverArt': u'234823794',
|
|
u'id': u'238472893',
|
|
u'isDir': True,
|
|
u'parent': u'2308472938',
|
|
u'title': u'Midnight Marauders'},
|
|
{u'artist': u'A Tribe Called Quest',
|
|
u'coverArt': u'39284792374',
|
|
u'id': u'983274892',
|
|
u'isDir': True,
|
|
u'parent': u'9823749',
|
|
u'title': u"People's Instinctive Travels And The Paths Of Rhythm"},
|
|
{u'artist': u'A Tribe Called Quest',
|
|
u'coverArt': u'289347293',
|
|
u'id': u'3894723934',
|
|
u'isDir': True,
|
|
u'parent': u'9832942',
|
|
u'title': u'The Anthology'},
|
|
{u'artist': u'A Tribe Called Quest',
|
|
u'coverArt': u'923847923',
|
|
u'id': u'29834729',
|
|
u'isDir': True,
|
|
u'parent': u'2934872893',
|
|
u'title': u'The Love Movement'},
|
|
{u'artist': u'A Tribe Called Quest',
|
|
u'coverArt': u'9238742893',
|
|
u'id': u'238947293',
|
|
u'isDir': True,
|
|
u'parent': u'9432878492',
|
|
u'title': u'The Low End Theory'}],
|
|
u'id': u'329847293',
|
|
u'name': u'A Tribe Called Quest'},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getMusicDirectory'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName, {'id': mid})
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def search(self, artist=None, album=None, title=None, any=None,
|
|
count=20, offset=0, newerThan=None):
|
|
"""
|
|
since: 1.0.0
|
|
|
|
DEPRECATED SINCE API 1.4.0! USE search2() INSTEAD!
|
|
|
|
Returns a listing of files matching the given search criteria.
|
|
Supports paging with offset
|
|
|
|
artist:str Search for artist
|
|
album:str Search for album
|
|
title:str Search for title of song
|
|
any:str Search all fields
|
|
count:int Max number of results to return [default: 20]
|
|
offset:int Search result offset. For paging [default: 0]
|
|
newerThan:int Return matches newer than this timestamp
|
|
"""
|
|
if artist == album == title == any == None:
|
|
raise ArgumentError('Invalid search. You must supply search '
|
|
'criteria')
|
|
methodName = 'search'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'artist': artist, 'album': album,
|
|
'title': title, 'any': any, 'count': count, 'offset': offset,
|
|
'newerThan': self._ts2milli(newerThan)})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def search2(self, query, artistCount=20, artistOffset=0, albumCount=20,
|
|
albumOffset=0, songCount=20, songOffset=0, musicFolderId=None):
|
|
"""
|
|
since: 1.4.0
|
|
|
|
Returns albums, artists and songs matching the given search criteria.
|
|
Supports paging through the result.
|
|
|
|
query:str The search query
|
|
artistCount:int Max number of artists to return [default: 20]
|
|
artistOffset:int Search offset for artists (for paging) [default: 0]
|
|
albumCount:int Max number of albums to return [default: 20]
|
|
albumOffset:int Search offset for albums (for paging) [default: 0]
|
|
songCount:int Max number of songs to return [default: 20]
|
|
songOffset:int Search offset for songs (for paging) [default: 0]
|
|
musicFolderId:int Only return results from the music folder
|
|
with the given ID. See getMusicFolders
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'searchResult2': {u'album': [{u'artist': u'A Tribe Called Quest',
|
|
u'coverArt': u'289347',
|
|
u'id': u'32487298',
|
|
u'isDir': True,
|
|
u'parent': u'98374289',
|
|
u'title': u'The Love Movement'}],
|
|
u'artist': [{u'id': u'2947839',
|
|
u'name': u'A Tribe Called Quest'},
|
|
{u'id': u'239847239',
|
|
u'name': u'Tribe'}],
|
|
u'song': [{u'album': u'Beats, Rhymes And Life',
|
|
u'artist': u'A Tribe Called Quest',
|
|
u'bitRate': 224,
|
|
u'contentType': u'audio/mpeg',
|
|
u'coverArt': u'329847',
|
|
u'duration': 148,
|
|
u'genre': u'default',
|
|
u'id': u'3928472893',
|
|
u'isDir': False,
|
|
u'isVideo': False,
|
|
u'parent': u'23984728394',
|
|
u'path': u'A Tribe Called Quest/Beats, Rhymes And Life/A Tribe Called Quest - Beats, Rhymes And Life - 03 - Motivators.mp3',
|
|
u'size': 4171913,
|
|
u'suffix': u'mp3',
|
|
u'title': u'Motivators',
|
|
u'track': 3}]},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'search2'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'query': query, 'artistCount': artistCount,
|
|
'artistOffset': artistOffset, 'albumCount': albumCount,
|
|
'albumOffset': albumOffset, 'songCount': songCount,
|
|
'songOffset': songOffset, 'musicFolderId': musicFolderId})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def search3(self, query, artistCount=20, artistOffset=0, albumCount=20,
|
|
albumOffset=0, songCount=20, songOffset=0, musicFolderId=None):
|
|
"""
|
|
since: 1.8.0
|
|
|
|
Works the same way as search2, but uses ID3 tags for
|
|
organization
|
|
|
|
query:str The search query
|
|
artistCount:int Max number of artists to return [default: 20]
|
|
artistOffset:int Search offset for artists (for paging) [default: 0]
|
|
albumCount:int Max number of albums to return [default: 20]
|
|
albumOffset:int Search offset for albums (for paging) [default: 0]
|
|
songCount:int Max number of songs to return [default: 20]
|
|
songOffset:int Search offset for songs (for paging) [default: 0]
|
|
musicFolderId:int Only return results from the music folder
|
|
with the given ID. See getMusicFolders
|
|
|
|
Returns a dict like the following (search for "Tune Yards":
|
|
{u'searchResult3': {u'album': [{u'artist': u'Tune-Yards',
|
|
u'artistId': 1,
|
|
u'coverArt': u'al-7',
|
|
u'created': u'2012-01-30T12:35:33',
|
|
u'duration': 3229,
|
|
u'id': 7,
|
|
u'name': u'Bird-Brains',
|
|
u'songCount': 13},
|
|
{u'artist': u'Tune-Yards',
|
|
u'artistId': 1,
|
|
u'coverArt': u'al-8',
|
|
u'created': u'2011-03-22T15:08:00',
|
|
u'duration': 2531,
|
|
u'id': 8,
|
|
u'name': u'W H O K I L L',
|
|
u'songCount': 10}],
|
|
u'artist': {u'albumCount': 2,
|
|
u'coverArt': u'ar-1',
|
|
u'id': 1,
|
|
u'name': u'Tune-Yards'},
|
|
u'song': [{u'album': u'Bird-Brains',
|
|
u'albumId': 7,
|
|
u'artist': u'Tune-Yards',
|
|
u'artistId': 1,
|
|
u'bitRate': 160,
|
|
u'contentType': u'audio/mpeg',
|
|
u'coverArt': 105,
|
|
u'created': u'2012-01-30T12:35:33',
|
|
u'duration': 328,
|
|
u'genre': u'Lo-Fi',
|
|
u'id': 107,
|
|
u'isDir': False,
|
|
u'isVideo': False,
|
|
u'parent': 105,
|
|
u'path': u'Tune Yards/Bird-Brains/10-tune-yards-fiya.mp3',
|
|
u'size': 6588498,
|
|
u'suffix': u'mp3',
|
|
u'title': u'Fiya',
|
|
u'track': 10,
|
|
u'type': u'music',
|
|
u'year': 2009}]},
|
|
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'search3'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'query': query, 'artistCount': artistCount,
|
|
'artistOffset': artistOffset, 'albumCount': albumCount,
|
|
'albumOffset': albumOffset, 'songCount': songCount,
|
|
'songOffset': songOffset, 'musicFolderId': musicFolderId})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getPlaylists(self, username=None):
|
|
"""
|
|
since: 1.0.0
|
|
|
|
Returns the ID and name of all saved playlists
|
|
The "username" option was added in 1.8.0.
|
|
|
|
username:str If specified, return playlists for this user
|
|
rather than for the authenticated user. The
|
|
authenticated user must have admin role
|
|
if this parameter is used
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'playlists': {u'playlist': [{u'id': u'62656174732e6d3375',
|
|
u'name': u'beats'},
|
|
{u'id': u'766172696574792e6d3375',
|
|
u'name': u'variety'}]},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getPlaylists'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'username': username})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getPlaylist(self, pid):
|
|
"""
|
|
since: 1.0.0
|
|
|
|
Returns a listing of files in a saved playlist
|
|
|
|
id:str The ID of the playlist as returned in getPlaylists()
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'playlist': {u'entry': {u'album': u'The Essential Bob Dylan',
|
|
u'artist': u'Bob Dylan',
|
|
u'bitRate': 32,
|
|
u'contentType': u'audio/mpeg',
|
|
u'coverArt': u'2983478293',
|
|
u'duration': 984,
|
|
u'genre': u'Classic Rock',
|
|
u'id': u'982739428',
|
|
u'isDir': False,
|
|
u'isVideo': False,
|
|
u'parent': u'98327428974',
|
|
u'path': u"Bob Dylan/Essential Bob Dylan Disc 1/Bob Dylan - The Essential Bob Dylan - 03 - The Times They Are A-Changin'.mp3",
|
|
u'size': 3921899,
|
|
u'suffix': u'mp3',
|
|
u'title': u"The Times They Are A-Changin'",
|
|
u'track': 3},
|
|
u'id': u'44796c616e2e6d3375',
|
|
u'name': u'Dylan'},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getPlaylist'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName, {'id': pid})
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def createPlaylist(self, playlistId=None, name=None, songIds=[]):
|
|
"""
|
|
since: 1.2.0
|
|
|
|
Creates OR updates a playlist. If updating the list, the
|
|
playlistId is required. If creating a list, the name is required.
|
|
|
|
playlistId:str The ID of the playlist to UPDATE
|
|
name:str The name of the playlist to CREATE
|
|
songIds:list The list of songIds to populate the list with in
|
|
either create or update mode. Note that this
|
|
list will replace the existing list if updating
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'createPlaylist'
|
|
viewName = '%s.view' % methodName
|
|
|
|
if playlistId == name == None:
|
|
raise ArgumentError('You must supply either a playlistId or a name')
|
|
if playlistId is not None and name is not None:
|
|
raise ArgumentError('You can only supply either a playlistId '
|
|
'OR a name, not both')
|
|
|
|
q = self._getQueryDict({'playlistId': playlistId, 'name': name})
|
|
|
|
req = self._getRequestWithList(viewName, 'songId', songIds, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def deletePlaylist(self, pid):
|
|
"""
|
|
since: 1.2.0
|
|
|
|
Deletes a saved playlist
|
|
|
|
pid:str ID of the playlist to delete, as obtained by getPlaylists
|
|
|
|
Returns a dict like the following:
|
|
|
|
"""
|
|
methodName = 'deletePlaylist'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName, {'id': pid})
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def download(self, sid):
|
|
"""
|
|
since: 1.0.0
|
|
|
|
Downloads a given music file.
|
|
|
|
sid:str The ID of the music file to download.
|
|
|
|
Returns the file-like object for reading or raises an exception
|
|
on error
|
|
"""
|
|
methodName = 'download'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName, {'id': sid})
|
|
res = self._doBinReq(req)
|
|
if isinstance(res, dict):
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def stream(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)
|
|
res = self._doBinReq(req)
|
|
if isinstance(res, dict):
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getCoverArt(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 res
|
|
|
|
def scrobble(self, sid, submission=True, listenTime=None):
|
|
"""
|
|
since: 1.5.0
|
|
|
|
"Scrobbles" a given music file on last.fm. Requires that the user
|
|
has set this up.
|
|
|
|
Since 1.8.0 you may specify multiple id (and optionally time)
|
|
parameters to scrobble multiple files.
|
|
|
|
Since 1.11.0 this method will also update the play count and
|
|
last played timestamp for the song and album. It will also make
|
|
the song appear in the "Now playing" page in the web app, and
|
|
appear in the list of songs returned by getNowPlaying
|
|
|
|
sid:str The ID of the file to scrobble
|
|
submission:bool Whether this is a "submission" or a "now playing"
|
|
notification
|
|
listenTime:int (Since 1.8.0) The time (unix timestamp) at
|
|
which the song was listened to.
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'scrobble'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'id': sid, 'submission': submission,
|
|
'time': self._ts2milli(listenTime)})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def changePassword(self, username, password):
|
|
"""
|
|
since: 1.1.0
|
|
|
|
Changes the password of an existing Subsonic user. Note that the
|
|
user performing this must have admin privileges
|
|
|
|
username:str The username whose password is being changed
|
|
password:str The new password of the user
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'changePassword'
|
|
viewName = '%s.view' % methodName
|
|
hexPass = 'enc:%s' % self._hexEnc(password)
|
|
|
|
# There seems to be an issue with some subsonic implementations
|
|
# not recognizing the "enc:" precursor to the encoded password and
|
|
# encodes the whole "enc:<hex>" as the password. Weird.
|
|
#q = {'username': username, 'password': hexPass.lower()}
|
|
q = {'username': username, 'password': password}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getUser(self, username):
|
|
"""
|
|
since: 1.3.0
|
|
|
|
Get details about a given user, including which auth roles it has.
|
|
Can be used to enable/disable certain features in the client, such
|
|
as jukebox control
|
|
|
|
username:str The username to retrieve. You can only retrieve
|
|
your own user unless you have admin privs.
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'status': u'ok',
|
|
u'user': {u'adminRole': False,
|
|
u'commentRole': False,
|
|
u'coverArtRole': False,
|
|
u'downloadRole': True,
|
|
u'jukeboxRole': False,
|
|
u'playlistRole': True,
|
|
u'podcastRole': False,
|
|
u'settingsRole': True,
|
|
u'streamRole': True,
|
|
u'uploadRole': True,
|
|
u'username': u'test'},
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getUser'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'username': username}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getUsers(self):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Gets a list of users
|
|
|
|
returns a dict like the following
|
|
|
|
{u'status': u'ok',
|
|
u'users': {u'user': [{u'adminRole': True,
|
|
u'commentRole': True,
|
|
u'coverArtRole': True,
|
|
u'downloadRole': True,
|
|
u'jukeboxRole': True,
|
|
u'playlistRole': True,
|
|
u'podcastRole': True,
|
|
u'scrobblingEnabled': True,
|
|
u'settingsRole': True,
|
|
u'shareRole': True,
|
|
u'streamRole': True,
|
|
u'uploadRole': True,
|
|
u'username': u'user1'},
|
|
...
|
|
...
|
|
]},
|
|
u'version': u'1.10.2',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getUsers'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def createUser(self, username, password, email,
|
|
ldapAuthenticated=False, adminRole=False, settingsRole=True,
|
|
streamRole=True, jukeboxRole=False, downloadRole=False,
|
|
uploadRole=False, playlistRole=False, coverArtRole=False,
|
|
commentRole=False, podcastRole=False, shareRole=False,
|
|
musicFolderId=None):
|
|
"""
|
|
since: 1.1.0
|
|
|
|
Creates a new subsonic user, using the parameters defined. See the
|
|
documentation at http://subsonic.org for more info on all the roles.
|
|
|
|
username:str The username of the new user
|
|
password:str The password for the new user
|
|
email:str The email of the new user
|
|
<For info on the boolean roles, see http://subsonic.org for more info>
|
|
musicFolderId:int These are the only folders the user has access to
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'createUser'
|
|
viewName = '%s.view' % methodName
|
|
hexPass = 'enc:%s' % self._hexEnc(password)
|
|
|
|
q = self._getQueryDict({
|
|
'username': username, 'password': hexPass, 'email': email,
|
|
'ldapAuthenticated': ldapAuthenticated, 'adminRole': adminRole,
|
|
'settingsRole': settingsRole, 'streamRole': streamRole,
|
|
'jukeboxRole': jukeboxRole, 'downloadRole': downloadRole,
|
|
'uploadRole': uploadRole, 'playlistRole': playlistRole,
|
|
'coverArtRole': coverArtRole, 'commentRole': commentRole,
|
|
'podcastRole': podcastRole, 'shareRole': shareRole,
|
|
'musicFolderId': musicFolderId
|
|
})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def updateUser(self, username, password=None, email=None,
|
|
ldapAuthenticated=False, adminRole=False, settingsRole=True,
|
|
streamRole=True, jukeboxRole=False, downloadRole=False,
|
|
uploadRole=False, playlistRole=False, coverArtRole=False,
|
|
commentRole=False, podcastRole=False, shareRole=False,
|
|
musicFolderId=None, maxBitRate=0):
|
|
"""
|
|
since 1.10.1
|
|
|
|
Modifies an existing Subsonic user.
|
|
|
|
username:str The username of the user to update.
|
|
musicFolderId:int Only return results from the music folder
|
|
with the given ID. See getMusicFolders
|
|
maxBitRate:int The max bitrate for the user. 0 is unlimited
|
|
|
|
All other args are the same as create user and you can update
|
|
whatever item you wish to update for the given username.
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'updateUser'
|
|
viewName = '%s.view' % methodName
|
|
if password is not None:
|
|
password = 'enc:%s' % self._hexEnc(password)
|
|
q = self._getQueryDict({'username': username, 'password': password,
|
|
'email': email, 'ldapAuthenticated': ldapAuthenticated,
|
|
'adminRole': adminRole,
|
|
'settingsRole': settingsRole, 'streamRole': streamRole,
|
|
'jukeboxRole': jukeboxRole, 'downloadRole': downloadRole,
|
|
'uploadRole': uploadRole, 'playlistRole': playlistRole,
|
|
'coverArtRole': coverArtRole, 'commentRole': commentRole,
|
|
'podcastRole': podcastRole, 'shareRole': shareRole,
|
|
'musicFolderId': musicFolderId, 'maxBitRate': maxBitRate
|
|
})
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def deleteUser(self, username):
|
|
"""
|
|
since: 1.3.0
|
|
|
|
Deletes an existing Subsonic user. Of course, you must have admin
|
|
rights for this.
|
|
|
|
username:str The username of the user to delete
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'deleteUser'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'username': username}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getChatMessages(self, since=1):
|
|
"""
|
|
since: 1.2.0
|
|
|
|
Returns the current visible (non-expired) chat messages.
|
|
|
|
since:int Only return messages newer than this timestamp
|
|
|
|
NOTE: All times returned are in MILLISECONDS since the Epoch, not
|
|
seconds!
|
|
|
|
Returns a dict like the following:
|
|
{u'chatMessages': {u'chatMessage': {u'message': u'testing 123',
|
|
u'time': 1303411919872L,
|
|
u'username': u'admin'}},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getChatMessages'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'since': self._ts2milli(since)}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def addChatMessage(self, message):
|
|
"""
|
|
since: 1.2.0
|
|
|
|
Adds a message to the chat log
|
|
|
|
message:str The message to add
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'addChatMessage'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'message': message}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getAlbumList(self, ltype, size=10, offset=0, fromYear=None,
|
|
toYear=None, genre=None, musicFolderId=None):
|
|
"""
|
|
since: 1.2.0
|
|
|
|
Returns a list of random, newest, highest rated etc. albums.
|
|
Similar to the album lists on the home page of the Subsonic
|
|
web interface
|
|
|
|
ltype:str The list type. Must be one of the following: random,
|
|
newest, highest, frequent, recent,
|
|
(since 1.8.0 -> )starred, alphabeticalByName,
|
|
alphabeticalByArtist
|
|
Since 1.10.1 you can use byYear and byGenre to
|
|
list albums in a given year range or genre.
|
|
size:int The number of albums to return. Max 500
|
|
offset:int The list offset. Use for paging. Max 5000
|
|
fromYear:int If you specify the ltype as "byYear", you *must*
|
|
specify fromYear
|
|
toYear:int If you specify the ltype as "byYear", you *must*
|
|
specify toYear
|
|
genre:str The name of the genre e.g. "Rock". You must specify
|
|
genre if you set the ltype to "byGenre"
|
|
musicFolderId:str Only return albums in the music folder with
|
|
the given ID. See getMusicFolders()
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'albumList': {u'album': [{u'artist': u'Hank Williams',
|
|
u'id': u'3264928374',
|
|
u'isDir': True,
|
|
u'parent': u'9238479283',
|
|
u'title': u'The Original Singles Collection...Plus'},
|
|
{u'artist': u'Freundeskreis',
|
|
u'coverArt': u'9823749823',
|
|
u'id': u'23492834',
|
|
u'isDir': True,
|
|
u'parent': u'9827492374',
|
|
u'title': u'Quadratur des Kreises'}]},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getAlbumList'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'type': ltype, 'size': size,
|
|
'offset': offset, 'fromYear': fromYear, 'toYear': toYear,
|
|
'genre': genre, 'musicFolderId': musicFolderId})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getAlbumList2(self, ltype, size=10, offset=0, fromYear=None,
|
|
toYear=None, genre=None):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Returns a list of random, newest, highest rated etc. albums.
|
|
This is similar to getAlbumList, but uses ID3 tags for
|
|
organization
|
|
|
|
ltype:str The list type. Must be one of the following: random,
|
|
newest, highest, frequent, recent,
|
|
(since 1.8.0 -> )starred, alphabeticalByName,
|
|
alphabeticalByArtist
|
|
Since 1.10.1 you can use byYear and byGenre to
|
|
list albums in a given year range or genre.
|
|
size:int The number of albums to return. Max 500
|
|
offset:int The list offset. Use for paging. Max 5000
|
|
fromYear:int If you specify the ltype as "byYear", you *must*
|
|
specify fromYear
|
|
toYear:int If you specify the ltype as "byYear", you *must*
|
|
specify toYear
|
|
genre:str The name of the genre e.g. "Rock". You must specify
|
|
genre if you set the ltype to "byGenre"
|
|
|
|
Returns a dict like the following:
|
|
{u'albumList2': {u'album': [{u'artist': u'Massive Attack',
|
|
u'artistId': 0,
|
|
u'coverArt': u'al-0',
|
|
u'created': u'2009-08-28T10:00:44',
|
|
u'duration': 3762,
|
|
u'id': 0,
|
|
u'name': u'100th Window',
|
|
u'songCount': 9},
|
|
{u'artist': u'Massive Attack',
|
|
u'artistId': 0,
|
|
u'coverArt': u'al-5',
|
|
u'created': u'2003-11-03T22:00:00',
|
|
u'duration': 2715,
|
|
u'id': 5,
|
|
u'name': u'Blue Lines',
|
|
u'songCount': 9}]},
|
|
u'status': u'ok',
|
|
u'version': u'1.8.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getAlbumList2'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'type': ltype, 'size': size,
|
|
'offset': offset, 'fromYear': fromYear, 'toYear': toYear,
|
|
'genre': genre})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getRandomSongs(self, size=10, genre=None, fromYear=None,
|
|
toYear=None, musicFolderId=None):
|
|
"""
|
|
since 1.2.0
|
|
|
|
Returns random songs matching the given criteria
|
|
|
|
size:int The max number of songs to return. Max 500
|
|
genre:str Only return songs from this genre
|
|
fromYear:int Only return songs after or in this year
|
|
toYear:int Only return songs before or in this year
|
|
musicFolderId:str Only return songs in the music folder with the
|
|
given ID. See getMusicFolders
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'randomSongs': {u'song': [{u'album': u'1998 EP - Airbag (How Am I Driving)',
|
|
u'artist': u'Radiohead',
|
|
u'bitRate': 320,
|
|
u'contentType': u'audio/mpeg',
|
|
u'duration': 129,
|
|
u'id': u'9284728934',
|
|
u'isDir': False,
|
|
u'isVideo': False,
|
|
u'parent': u'983249823',
|
|
u'path': u'Radiohead/1998 EP - Airbag (How Am I Driving)/06 - Melatonin.mp3',
|
|
u'size': 5177469,
|
|
u'suffix': u'mp3',
|
|
u'title': u'Melatonin'},
|
|
{u'album': u'Mezmerize',
|
|
u'artist': u'System Of A Down',
|
|
u'bitRate': 214,
|
|
u'contentType': u'audio/mpeg',
|
|
u'coverArt': u'23849372894',
|
|
u'duration': 176,
|
|
u'id': u'28937492834',
|
|
u'isDir': False,
|
|
u'isVideo': False,
|
|
u'parent': u'92837492837',
|
|
u'path': u'System Of A Down/Mesmerize/10 - System Of A Down - Old School Hollywood.mp3',
|
|
u'size': 4751360,
|
|
u'suffix': u'mp3',
|
|
u'title': u'Old School Hollywood',
|
|
u'track': 10}]},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getRandomSongs'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'size': size, 'genre': genre,
|
|
'fromYear': fromYear, 'toYear': toYear,
|
|
'musicFolderId': musicFolderId})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getLyrics(self, artist=None, title=None):
|
|
"""
|
|
since: 1.2.0
|
|
|
|
Searches for and returns lyrics for a given song
|
|
|
|
artist:str The artist name
|
|
title:str The song title
|
|
|
|
Returns a dict like the following for
|
|
getLyrics('Bob Dylan', 'Blowin in the wind'):
|
|
|
|
{u'lyrics': {u'artist': u'Bob Dylan',
|
|
u'content': u"How many roads must a man walk down<snip>",
|
|
u'title': u"Blowin' in the Wind"},
|
|
u'status': u'ok',
|
|
u'version': u'1.5.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getLyrics'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'artist': artist, 'title': title})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def jukeboxControl(self, action, index=None, sids=[], gain=None,
|
|
offset=None):
|
|
"""
|
|
since: 1.2.0
|
|
|
|
NOTE: Some options were added as of API version 1.7.0
|
|
|
|
Controls the jukebox, i.e., playback directly on the server's
|
|
audio hardware. Note: The user must be authorized to control
|
|
the jukebox
|
|
|
|
action:str The operation to perform. Must be one of: get,
|
|
start, stop, skip, add, clear, remove, shuffle,
|
|
setGain, status (added in API 1.7.0),
|
|
set (added in API 1.7.0)
|
|
index:int Used by skip and remove. Zero-based index of the
|
|
song to skip to or remove.
|
|
sids:str Used by "add" and "set". ID of song to add to the
|
|
jukebox playlist. Use multiple id parameters to
|
|
add many songs in the same request. Whether you
|
|
are passing one song or many into this, this
|
|
parameter MUST be a list
|
|
gain:float Used by setGain to control the playback volume.
|
|
A float value between 0.0 and 1.0
|
|
offset:int (added in API 1.7.0) Used by "skip". Start playing
|
|
this many seconds into the track.
|
|
"""
|
|
methodName = 'jukeboxControl'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'action': action, 'index': index,
|
|
'gain': gain, 'offset': offset})
|
|
|
|
req = None
|
|
if action == 'add':
|
|
# We have to deal with the sids
|
|
if not (isinstance(sids, list) or isinstance(sids, tuple)):
|
|
raise ArgumentError('If you are adding songs, "sids" must '
|
|
'be a list or tuple!')
|
|
req = self._getRequestWithList(viewName, 'id', sids, q)
|
|
else:
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getPodcasts(self, incEpisodes=True, pid=None):
|
|
"""
|
|
since: 1.6.0
|
|
|
|
Returns all podcast channels the server subscribes to and their
|
|
episodes.
|
|
|
|
incEpisodes:bool (since: 1.9.0) Whether to include Podcast
|
|
episodes in the returned result.
|
|
pid:str (since: 1.9.0) If specified, only return
|
|
the Podcast channel with this ID.
|
|
|
|
Returns a dict like the following:
|
|
{u'status': u'ok',
|
|
u'version': u'1.6.0',
|
|
u'xmlns': u'http://subsonic.org/restapi',
|
|
u'podcasts': {u'channel': {u'description': u"Dr Chris Smith...",
|
|
u'episode': [{u'album': u'Dr Karl and the Naked Scientist',
|
|
u'artist': u'BBC Radio 5 live',
|
|
u'bitRate': 64,
|
|
u'contentType': u'audio/mpeg',
|
|
u'coverArt': u'2f6f7074',
|
|
u'description': u'Dr Karl answers all your science related questions.',
|
|
u'duration': 2902,
|
|
u'genre': u'Podcast',
|
|
u'id': 0,
|
|
u'isDir': False,
|
|
u'isVideo': False,
|
|
u'parent': u'2f6f70742f737562736f6e69632f706f6463617374732f4472204b61726c20616e6420746865204e616b656420536369656e74697374',
|
|
u'publishDate': u'2011-08-17 22:06:00.0',
|
|
u'size': 23313059,
|
|
u'status': u'completed',
|
|
u'streamId': u'2f6f70742f737562736f6e69632f706f6463617374732f4472204b61726c20616e6420746865204e616b656420536369656e746973742f64726b61726c5f32303131303831382d30343036612e6d7033',
|
|
u'suffix': u'mp3',
|
|
u'title': u'DrKarl: Peppermints, Chillies & Receptors',
|
|
u'year': 2011},
|
|
{u'description': u'which is warmer, a bath with bubbles in it or one without? Just one of the stranger science stories tackled this week by Dr Chris Smith and the Naked Scientists!',
|
|
u'id': 1,
|
|
u'publishDate': u'2011-08-14 21:05:00.0',
|
|
u'status': u'skipped',
|
|
u'title': u'DrKarl: how many bubbles in your bath? 15 AUG 11'},
|
|
...
|
|
{u'description': u'Dr Karl joins Rhod to answer all your science questions',
|
|
u'id': 9,
|
|
u'publishDate': u'2011-07-06 22:12:00.0',
|
|
u'status': u'skipped',
|
|
u'title': u'DrKarl: 8 Jul 11 The Strange Sound of the MRI Scanner'}],
|
|
u'id': 0,
|
|
u'status': u'completed',
|
|
u'title': u'Dr Karl and the Naked Scientist',
|
|
u'url': u'http://downloads.bbc.co.uk/podcasts/fivelive/drkarl/rss.xml'}}
|
|
}
|
|
|
|
See also: http://subsonic.svn.sourceforge.net/viewvc/subsonic/trunk/subsonic-main/src/main/webapp/xsd/podcasts_example_1.xml?view=markup
|
|
"""
|
|
methodName = 'getPodcasts'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'includeEpisodes': incEpisodes,
|
|
'id': pid})
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getShares(self):
|
|
"""
|
|
since: 1.6.0
|
|
|
|
Returns information about shared media this user is allowed to manage
|
|
|
|
Note that entry can be either a single dict or a list of dicts
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'status': u'ok',
|
|
u'version': u'1.6.0',
|
|
u'xmlns': u'http://subsonic.org/restapi',
|
|
u'shares': {u'share': [
|
|
{u'created': u'2011-08-18T10:01:35',
|
|
u'entry': {u'artist': u'Alice In Chains',
|
|
u'coverArt': u'2f66696c65732f6d7033732f412d4d2f416c69636520496e20436861696e732f416c69636520496e20436861696e732f636f7665722e6a7067',
|
|
u'id': u'2f66696c65732f6d7033732f412d4d2f416c69636520496e20436861696e732f416c69636520496e20436861696e73',
|
|
u'isDir': True,
|
|
u'parent': u'2f66696c65732f6d7033732f412d4d2f416c69636520496e20436861696e73',
|
|
u'title': u'Alice In Chains'},
|
|
u'expires': u'2012-08-18T10:01:35',
|
|
u'id': 0,
|
|
u'url': u'http://crustymonkey.subsonic.org/share/BuLbF',
|
|
u'username': u'admin',
|
|
u'visitCount': 0
|
|
}]}
|
|
}
|
|
"""
|
|
methodName = 'getShares'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def createShare(self, shids=[], description=None, expires=None):
|
|
"""
|
|
since: 1.6.0
|
|
|
|
Creates a public URL that can be used by anyone to stream music
|
|
or video from the Subsonic server. The URL is short and suitable
|
|
for posting on Facebook, Twitter etc. Note: The user must be
|
|
authorized to share (see Settings > Users > User is allowed to
|
|
share files with anyone).
|
|
|
|
shids:list[str] A list of ids of songs, albums or videos
|
|
to share.
|
|
description:str A description that will be displayed to
|
|
people visiting the shared media
|
|
(optional).
|
|
expires:float A timestamp pertaining to the time at
|
|
which this should expire (optional)
|
|
|
|
This returns a structure like you would get back from getShares()
|
|
containing just your new share.
|
|
"""
|
|
methodName = 'createShare'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'description': description,
|
|
'expires': self._ts2milli(expires)})
|
|
req = self._getRequestWithList(viewName, 'id', shids, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def updateShare(self, shid, description=None, expires=None):
|
|
"""
|
|
since: 1.6.0
|
|
|
|
Updates the description and/or expiration date for an existing share
|
|
|
|
shid:str The id of the share to update
|
|
description:str The new description for the share (optional).
|
|
expires:float The new timestamp for the expiration time of this
|
|
share (optional).
|
|
"""
|
|
methodName = 'updateShare'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'id': shid, 'description': description,
|
|
expires: self._ts2milli(expires)})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def deleteShare(self, shid):
|
|
"""
|
|
since: 1.6.0
|
|
|
|
Deletes an existing share
|
|
|
|
shid:str The id of the share to delete
|
|
|
|
Returns a standard response dict
|
|
"""
|
|
methodName = 'deleteShare'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'id': shid})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def setRating(self, id, rating):
|
|
"""
|
|
since: 1.6.0
|
|
|
|
Sets the rating for a music file
|
|
|
|
id:str The id of the item (song/artist/album) to rate
|
|
rating:int The rating between 1 and 5 (inclusive), or 0 to remove
|
|
the rating
|
|
|
|
Returns a standard response dict
|
|
"""
|
|
methodName = 'setRating'
|
|
viewName = '%s.view' % methodName
|
|
|
|
try:
|
|
rating = int(rating)
|
|
except:
|
|
raise ArgumentError('Rating must be an integer between 0 and 5: '
|
|
'%r' % rating)
|
|
if rating < 0 or rating > 5:
|
|
raise ArgumentError('Rating must be an integer between 0 and 5: '
|
|
'%r' % rating)
|
|
|
|
q = self._getQueryDict({'id': id, 'rating': rating})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getArtists(self):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Similar to getIndexes(), but this method uses the ID3 tags to
|
|
determine the artist
|
|
|
|
Returns a dict like the following:
|
|
{u'artists': {u'index': [{u'artist': {u'albumCount': 7,
|
|
u'coverArt': u'ar-0',
|
|
u'id': 0,
|
|
u'name': u'Massive Attack'},
|
|
u'name': u'M'},
|
|
{u'artist': {u'albumCount': 2,
|
|
u'coverArt': u'ar-1',
|
|
u'id': 1,
|
|
u'name': u'Tune-Yards'},
|
|
u'name': u'T'}]},
|
|
u'status': u'ok',
|
|
u'version': u'1.8.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getArtists'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getArtist(self, id):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Returns the info (albums) for an artist. This method uses
|
|
the ID3 tags for organization
|
|
|
|
id:str The artist ID
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'artist': {u'album': [{u'artist': u'Tune-Yards',
|
|
u'artistId': 1,
|
|
u'coverArt': u'al-7',
|
|
u'created': u'2012-01-30T12:35:33',
|
|
u'duration': 3229,
|
|
u'id': 7,
|
|
u'name': u'Bird-Brains',
|
|
u'songCount': 13},
|
|
{u'artist': u'Tune-Yards',
|
|
u'artistId': 1,
|
|
u'coverArt': u'al-8',
|
|
u'created': u'2011-03-22T15:08:00',
|
|
u'duration': 2531,
|
|
u'id': 8,
|
|
u'name': u'W H O K I L L',
|
|
u'songCount': 10}],
|
|
u'albumCount': 2,
|
|
u'coverArt': u'ar-1',
|
|
u'id': 1,
|
|
u'name': u'Tune-Yards'},
|
|
u'status': u'ok',
|
|
u'version': u'1.8.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getArtist'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'id': id})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getAlbum(self, id):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Returns the info and songs for an album. This method uses
|
|
the ID3 tags for organization
|
|
|
|
id:str The album ID
|
|
|
|
Returns a dict like the following:
|
|
|
|
{u'album': {u'artist': u'Massive Attack',
|
|
u'artistId': 0,
|
|
u'coverArt': u'al-0',
|
|
u'created': u'2009-08-28T10:00:44',
|
|
u'duration': 3762,
|
|
u'id': 0,
|
|
u'name': u'100th Window',
|
|
u'song': [{u'album': u'100th Window',
|
|
u'albumId': 0,
|
|
u'artist': u'Massive Attack',
|
|
u'artistId': 0,
|
|
u'bitRate': 192,
|
|
u'contentType': u'audio/mpeg',
|
|
u'coverArt': 2,
|
|
u'created': u'2009-08-28T10:00:57',
|
|
u'duration': 341,
|
|
u'genre': u'Rock',
|
|
u'id': 14,
|
|
u'isDir': False,
|
|
u'isVideo': False,
|
|
u'parent': 2,
|
|
u'path': u'Massive Attack/100th Window/01 - Future Proof.mp3',
|
|
u'size': 8184445,
|
|
u'suffix': u'mp3',
|
|
u'title': u'Future Proof',
|
|
u'track': 1,
|
|
u'type': u'music',
|
|
u'year': 2003}],
|
|
u'songCount': 9},
|
|
u'status': u'ok',
|
|
u'version': u'1.8.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getAlbum'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'id': id})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getSong(self, id):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Returns the info for a song. This method uses the ID3
|
|
tags for organization
|
|
|
|
id:str The song ID
|
|
|
|
Returns a dict like the following:
|
|
{u'song': {u'album': u'W H O K I L L',
|
|
u'albumId': 8,
|
|
u'artist': u'Tune-Yards',
|
|
u'artistId': 1,
|
|
u'bitRate': 320,
|
|
u'contentType': u'audio/mpeg',
|
|
u'coverArt': 106,
|
|
u'created': u'2011-03-22T15:08:00',
|
|
u'discNumber': 1,
|
|
u'duration': 192,
|
|
u'genre': u'Indie Rock',
|
|
u'id': 120,
|
|
u'isDir': False,
|
|
u'isVideo': False,
|
|
u'parent': 106,
|
|
u'path': u'Tune Yards/Who Kill/10 Killa.mp3',
|
|
u'size': 7692656,
|
|
u'suffix': u'mp3',
|
|
u'title': u'Killa',
|
|
u'track': 10,
|
|
u'type': u'music',
|
|
u'year': 2011},
|
|
u'status': u'ok',
|
|
u'version': u'1.8.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getSong'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'id': id})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getVideos(self):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Returns all video files
|
|
|
|
Returns a dict like the following:
|
|
{u'status': u'ok',
|
|
u'version': u'1.8.0',
|
|
u'videos': {u'video': {u'bitRate': 384,
|
|
u'contentType': u'video/x-matroska',
|
|
u'created': u'2012-08-26T13:36:44',
|
|
u'duration': 1301,
|
|
u'id': 130,
|
|
u'isDir': False,
|
|
u'isVideo': True,
|
|
u'path': u'South Park - 16x07 - Cartman Finds Love.mkv',
|
|
u'size': 287309613,
|
|
u'suffix': u'mkv',
|
|
u'title': u'South Park - 16x07 - Cartman Finds Love',
|
|
u'transcodedContentType': u'video/x-flv',
|
|
u'transcodedSuffix': u'flv'}},
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getVideos'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getStarred(self, musicFolderId=None):
|
|
"""
|
|
since 1.8.0
|
|
|
|
musicFolderId:int Only return results from the music folder
|
|
with the given ID. See getMusicFolders
|
|
|
|
Returns starred songs, albums and artists
|
|
|
|
Returns a dict like the following:
|
|
{u'starred': {u'album': {u'album': u'Bird-Brains',
|
|
u'artist': u'Tune-Yards',
|
|
u'coverArt': 105,
|
|
u'created': u'2012-01-30T13:16:58',
|
|
u'id': 105,
|
|
u'isDir': True,
|
|
u'parent': 104,
|
|
u'starred': u'2012-08-26T13:18:34',
|
|
u'title': u'Bird-Brains'},
|
|
u'song': [{u'album': u'Mezzanine',
|
|
u'albumId': 4,
|
|
u'artist': u'Massive Attack',
|
|
u'artistId': 0,
|
|
u'bitRate': 256,
|
|
u'contentType': u'audio/mpeg',
|
|
u'coverArt': 6,
|
|
u'created': u'2009-06-15T07:48:28',
|
|
u'duration': 298,
|
|
u'genre': u'Dub',
|
|
u'id': 72,
|
|
u'isDir': False,
|
|
u'isVideo': False,
|
|
u'parent': 6,
|
|
u'path': u'Massive Attack/Mezzanine/Massive Attack_02_mezzanine.mp3',
|
|
u'size': 9564160,
|
|
u'starred': u'2012-08-26T13:19:26',
|
|
u'suffix': u'mp3',
|
|
u'title': u'Risingson',
|
|
u'track': 2,
|
|
u'type': u'music'},
|
|
{u'album': u'Mezzanine',
|
|
u'albumId': 4,
|
|
u'artist': u'Massive Attack',
|
|
u'artistId': 0,
|
|
u'bitRate': 256,
|
|
u'contentType': u'audio/mpeg',
|
|
u'coverArt': 6,
|
|
u'created': u'2009-06-15T07:48:25',
|
|
u'duration': 380,
|
|
u'genre': u'Dub',
|
|
u'id': 71,
|
|
u'isDir': False,
|
|
u'isVideo': False,
|
|
u'parent': 6,
|
|
u'path': u'Massive Attack/Mezzanine/Massive Attack_01_mezzanine.mp3',
|
|
u'size': 12179456,
|
|
u'starred': u'2012-08-26T13:19:03',
|
|
u'suffix': u'mp3',
|
|
u'title': u'Angel',
|
|
u'track': 1,
|
|
u'type': u'music'}]},
|
|
u'status': u'ok',
|
|
u'version': u'1.8.0',
|
|
u'xmlns': u'http://subsonic.org/restapi'}
|
|
"""
|
|
methodName = 'getStarred'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {}
|
|
if musicFolderId:
|
|
q['musicFolderId'] = musicFolderId
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getStarred2(self, musicFolderId=None):
|
|
"""
|
|
since 1.8.0
|
|
|
|
musicFolderId:int Only return results from the music folder
|
|
with the given ID. See getMusicFolders
|
|
|
|
Returns starred songs, albums and artists like getStarred(),
|
|
but this uses ID3 tags for organization
|
|
|
|
Returns a dict like the following:
|
|
|
|
**See the output from getStarred()**
|
|
"""
|
|
methodName = 'getStarred2'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {}
|
|
if musicFolderId:
|
|
q['musicFolderId'] = musicFolderId
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def updatePlaylist(self, lid, name=None, comment=None, songIdsToAdd=[],
|
|
songIndexesToRemove=[]):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Updates a playlist. Only the owner of a playlist is allowed to
|
|
update it.
|
|
|
|
lid:str The playlist id
|
|
name:str The human readable name of the playlist
|
|
comment:str The playlist comment
|
|
songIdsToAdd:list A list of song IDs to add to the playlist
|
|
songIndexesToRemove:list Remove the songs at the
|
|
0 BASED INDEXED POSITIONS in the
|
|
playlist, NOT the song ids. Note that
|
|
this is always a list.
|
|
|
|
Returns a normal status response dict
|
|
"""
|
|
methodName = 'updatePlaylist'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'playlistId': lid, 'name': name,
|
|
'comment': comment})
|
|
if not isinstance(songIdsToAdd, list) or isinstance(songIdsToAdd,
|
|
tuple):
|
|
songIdsToAdd = [songIdsToAdd]
|
|
if not isinstance(songIndexesToRemove, list) or isinstance(
|
|
songIndexesToRemove, tuple):
|
|
songIndexesToRemove = [songIndexesToRemove]
|
|
listMap = {'songIdToAdd': songIdsToAdd,
|
|
'songIndexToRemove': songIndexesToRemove}
|
|
req = self._getRequestWithLists(viewName, listMap, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getAvatar(self, username):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Returns the avatar for a user or None if the avatar does not exist
|
|
|
|
username:str The user to retrieve the avatar for
|
|
|
|
Returns the file-like object for reading or raises an exception
|
|
on error
|
|
"""
|
|
methodName = 'getAvatar'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'username': username}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
try:
|
|
res = self._doBinReq(req)
|
|
except urllib2.HTTPError:
|
|
# Avatar is not set/does not exist, return None
|
|
return None
|
|
if isinstance(res, dict):
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def star(self, sids=[], albumIds=[], artistIds=[]):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Attaches a star to songs, albums or artists
|
|
|
|
sids:list A list of song IDs to star
|
|
albumIds:list A list of album IDs to star. Use this rather than
|
|
"sids" if the client access the media collection
|
|
according to ID3 tags rather than file
|
|
structure
|
|
artistIds:list The ID of an artist to star. Use this rather
|
|
than sids if the client access the media
|
|
collection according to ID3 tags rather
|
|
than file structure
|
|
|
|
Returns a normal status response dict
|
|
"""
|
|
methodName = 'star'
|
|
viewName = '%s.view' % methodName
|
|
|
|
if not isinstance(sids, list) or isinstance(sids, tuple):
|
|
sids = [sids]
|
|
if not isinstance(albumIds, list) or isinstance(albumIds, tuple):
|
|
albumIds = [albumIds]
|
|
if not isinstance(artistIds, list) or isinstance(artistIds, tuple):
|
|
artistIds = [artistIds]
|
|
listMap = {'id': sids,
|
|
'albumId': albumIds,
|
|
'artistId': artistIds}
|
|
req = self._getRequestWithLists(viewName, listMap)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def unstar(self, sids=[], albumIds=[], artistIds=[]):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Removes a star to songs, albums or artists. Basically, the
|
|
same as star in reverse
|
|
|
|
sids:list A list of song IDs to star
|
|
albumIds:list A list of album IDs to star. Use this rather than
|
|
"sids" if the client access the media collection
|
|
according to ID3 tags rather than file
|
|
structure
|
|
artistIds:list The ID of an artist to star. Use this rather
|
|
than sids if the client access the media
|
|
collection according to ID3 tags rather
|
|
than file structure
|
|
|
|
Returns a normal status response dict
|
|
"""
|
|
methodName = 'unstar'
|
|
viewName = '%s.view' % methodName
|
|
|
|
if not isinstance(sids, list) or isinstance(sids, tuple):
|
|
sids = [sids]
|
|
if not isinstance(albumIds, list) or isinstance(albumIds, tuple):
|
|
albumIds = [albumIds]
|
|
if not isinstance(artistIds, list) or isinstance(artistIds, tuple):
|
|
artistIds = [artistIds]
|
|
listMap = {'id': sids,
|
|
'albumId': albumIds,
|
|
'artistId': artistIds}
|
|
req = self._getRequestWithLists(viewName, listMap)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getGenres(self):
|
|
"""
|
|
since 1.9.0
|
|
|
|
Returns all genres
|
|
"""
|
|
methodName = 'getGenres'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getSongsByGenre(self, genre, count=10, offset=0, musicFolderId=None):
|
|
"""
|
|
since 1.9.0
|
|
|
|
Returns songs in a given genre
|
|
|
|
genre:str The genre, as returned by getGenres()
|
|
count:int The maximum number of songs to return. Max is 500
|
|
default: 10
|
|
offset:int The offset if you are paging. default: 0
|
|
musicFolderId:int Only return results from the music folder
|
|
with the given ID. See getMusicFolders
|
|
"""
|
|
methodName = 'getGenres'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'genre': genre,
|
|
'count': count,
|
|
'offset': offset,
|
|
'musicFolderId': musicFolderId,
|
|
})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def hls (self, mid, bitrate=None):
|
|
"""
|
|
since 1.8.0
|
|
|
|
Creates an HTTP live streaming playlist for streaming video or
|
|
audio HLS is a streaming protocol implemented by Apple and
|
|
works by breaking the overall stream into a sequence of small
|
|
HTTP-based file downloads. It's supported by iOS and newer
|
|
versions of Android. This method also supports adaptive
|
|
bitrate streaming, see the bitRate parameter.
|
|
|
|
mid:str The ID of the media to stream
|
|
bitrate:str If specified, the server will attempt to limit the
|
|
bitrate to this value, in kilobits per second. If
|
|
this parameter is specified more than once, the
|
|
server will create a variant playlist, suitable
|
|
for adaptive bitrate streaming. The playlist will
|
|
support streaming at all the specified bitrates.
|
|
The server will automatically choose video dimensions
|
|
that are suitable for the given bitrates.
|
|
(since: 1.9.0) you may explicitly request a certain
|
|
width (480) and height (360) like so:
|
|
bitRate=1000@480x360
|
|
|
|
Returns the raw m3u8 file as a string
|
|
"""
|
|
methodName = 'hls'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'id': mid, 'bitrate': bitrate})
|
|
req = self._getRequest(viewName, q)
|
|
try:
|
|
res = self._doBinReq(req)
|
|
except urllib2.HTTPError:
|
|
# Avatar is not set/does not exist, return None
|
|
return None
|
|
if isinstance(res, dict):
|
|
self._checkStatus(res)
|
|
return res.read()
|
|
|
|
def refreshPodcasts(self):
|
|
"""
|
|
since: 1.9.0
|
|
|
|
Tells the server to check for new Podcast episodes. Note: The user
|
|
must be authorized for Podcast administration
|
|
"""
|
|
methodName = 'refreshPodcasts'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def createPodcastChannel(self, url):
|
|
"""
|
|
since: 1.9.0
|
|
|
|
Adds a new Podcast channel. Note: The user must be authorized
|
|
for Podcast administration
|
|
|
|
url:str The URL of the Podcast to add
|
|
"""
|
|
methodName = 'createPodcastChannel'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'url': url}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def deletePodcastChannel(self, pid):
|
|
"""
|
|
since: 1.9.0
|
|
|
|
Deletes a Podcast channel. Note: The user must be authorized
|
|
for Podcast administration
|
|
|
|
pid:str The ID of the Podcast channel to delete
|
|
"""
|
|
methodName = 'deletePodcastChannel'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'id': pid}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def deletePodcastEpisode(self, pid):
|
|
"""
|
|
since: 1.9.0
|
|
|
|
Deletes a Podcast episode. Note: The user must be authorized
|
|
for Podcast administration
|
|
|
|
pid:str The ID of the Podcast episode to delete
|
|
"""
|
|
methodName = 'deletePodcastEpisode'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'id': pid}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def downloadPodcastEpisode(self, pid):
|
|
"""
|
|
since: 1.9.0
|
|
|
|
Tells the server to start downloading a given Podcast episode.
|
|
Note: The user must be authorized for Podcast administration
|
|
|
|
pid:str The ID of the Podcast episode to download
|
|
"""
|
|
methodName = 'downloadPodcastEpisode'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'id': pid}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getInternetRadioStations(self):
|
|
"""
|
|
since: 1.9.0
|
|
|
|
Returns all internet radio stations
|
|
"""
|
|
methodName = 'getInternetRadioStations'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getBookmarks(self):
|
|
"""
|
|
since: 1.9.0
|
|
|
|
Returns all bookmarks for this user. A bookmark is a position
|
|
within a media file
|
|
"""
|
|
methodName = 'getBookmarks'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def createBookmark(self, mid, position, comment=None):
|
|
"""
|
|
since: 1.9.0
|
|
|
|
Creates or updates a bookmark (position within a media file).
|
|
Bookmarks are personal and not visible to other users
|
|
|
|
mid:str The ID of the media file to bookmark. If a bookmark
|
|
already exists for this file, it will be overwritten
|
|
position:int The position (in milliseconds) within the media file
|
|
comment:str A user-defined comment
|
|
"""
|
|
methodName = 'createBookmark'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'id': mid, 'position': position,
|
|
'comment': comment})
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def deleteBookmark(self, mid):
|
|
"""
|
|
since: 1.9.0
|
|
|
|
Deletes the bookmark for a given file
|
|
|
|
mid:str The ID of the media file to delete the bookmark from.
|
|
Other users' bookmarks are not affected
|
|
"""
|
|
methodName = 'deleteBookmark'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'id': mid}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getArtistInfo(self, aid, count=20, includeNotPresent=False):
|
|
"""
|
|
since: 1.11.0
|
|
|
|
Returns artist info with biography, image URLS and similar artists
|
|
using data from last.fm
|
|
|
|
aid:str The ID of the artist, album or song
|
|
count:int The max number of similar artists to return
|
|
includeNotPresent:bool Whether to return artists that are not
|
|
present in the media library
|
|
"""
|
|
methodName = 'getArtistInfo'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'id': aid, 'count': count,
|
|
'includeNotPresent': includeNotPresent}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getArtistInfo2(self, aid, count=20, includeNotPresent=False):
|
|
"""
|
|
since: 1.11.0
|
|
|
|
Similar to getArtistInfo(), but organizes music according to ID3 tags
|
|
|
|
aid:str The ID of the artist, album or song
|
|
count:int The max number of similar artists to return
|
|
includeNotPresent:bool Whether to return artists that are not
|
|
present in the media library
|
|
"""
|
|
methodName = 'getArtistInfo2'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'id': aid, 'count': count,
|
|
'includeNotPresent': includeNotPresent}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getSimilarSongs(self, iid, count=50):
|
|
"""
|
|
since 1.11.0
|
|
|
|
Returns a random collection of songs from the given artist and
|
|
similar artists, using data from last.fm. Typically used for
|
|
artist radio features.
|
|
|
|
iid:str The artist, album, or song ID
|
|
count:int Max number of songs to return
|
|
"""
|
|
methodName = 'getSimilarSongs'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'id': iid, 'count': count}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getSimilarSongs2(self, iid, count=50):
|
|
"""
|
|
since 1.11.0
|
|
|
|
Similar to getSimilarSongs(), but organizes music according to
|
|
ID3 tags
|
|
|
|
iid:str The artist, album, or song ID
|
|
count:int Max number of songs to return
|
|
"""
|
|
methodName = 'getSimilarSongs2'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'id': iid, 'count': count}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def savePlayQueue(self, qids, current=None, position=None):
|
|
"""
|
|
since 1.12.0
|
|
|
|
qid:list[int] The list of song ids in the play queue
|
|
current:int The id of the current playing song
|
|
position:int The position, in milliseconds, within the current
|
|
playing song
|
|
|
|
Saves the state of the play queue for this user. This includes
|
|
the tracks in the play queue, the currently playing track, and
|
|
the position within this track. Typically used to allow a user to
|
|
move between different clients/apps while retaining the same play
|
|
queue (for instance when listening to an audio book).
|
|
"""
|
|
methodName = 'savePlayQueue'
|
|
viewName = '%s.view' % methodName
|
|
if not isinstance(qids, (tuple, list)):
|
|
qids = [qids]
|
|
|
|
q = self._getQueryDict({'current': current, 'position': position})
|
|
|
|
req = self._getRequestWithLists(viewName, {'id': qids}, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getPlayQueue(self):
|
|
"""
|
|
since 1.12.0
|
|
|
|
Returns the state of the play queue for this user (as set by
|
|
savePlayQueue). This includes the tracks in the play queue,
|
|
the currently playing track, and the position within this track.
|
|
Typically used to allow a user to move between different
|
|
clients/apps while retaining the same play queue (for instance
|
|
when listening to an audio book).
|
|
"""
|
|
methodName = 'getPlayQueue'
|
|
viewName = '%s.view' % methodName
|
|
|
|
req = self._getRequest(viewName)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getTopSongs(self, artist, count=50):
|
|
"""
|
|
since 1.13.0
|
|
|
|
Returns the top songs for a given artist
|
|
|
|
artist:str The artist to get songs for
|
|
count:int The number of songs to return
|
|
"""
|
|
methodName = 'getTopSongs'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'artist': artist, 'count': count}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getNewestPodcasts(self, count=20):
|
|
"""
|
|
since 1.13.0
|
|
|
|
Returns the most recently published Podcast episodes
|
|
|
|
count:int The number of episodes to return
|
|
"""
|
|
methodName = 'getNewestPodcasts'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'count': count}
|
|
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def scanMediaFolders(self):
|
|
"""
|
|
This is not an officially supported method of the API
|
|
|
|
Same as selecting 'Settings' > 'Scan media folders now' with
|
|
Subsonic web GUI
|
|
|
|
Returns True if refresh successful, False otherwise
|
|
"""
|
|
methodName = 'scanNow'
|
|
return self._unsupportedAPIFunction(methodName)
|
|
|
|
def cleanupDatabase(self):
|
|
"""
|
|
This is not an officially supported method of the API
|
|
|
|
Same as selecting 'Settings' > 'Clean-up Database' with Subsonic
|
|
web GUI
|
|
|
|
Returns True if cleanup initiated successfully, False otherwise
|
|
|
|
Subsonic stores information about all media files ever encountered.
|
|
By cleaning up the database, information about files that are
|
|
no longer in your media collection is permanently removed.
|
|
"""
|
|
methodName = 'expunge'
|
|
return self._unsupportedAPIFunction(methodName)
|
|
|
|
def getVideoInfo(self, vid):
|
|
"""
|
|
since 1.14.0
|
|
|
|
Returns details for a video, including information about available
|
|
audio tracks, subtitles (captions) and conversions.
|
|
|
|
vid:int The video ID
|
|
"""
|
|
methodName = 'getVideoInfo'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'id': int(vid)}
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getAlbumInfo(self, aid):
|
|
"""
|
|
since 1.14.0
|
|
|
|
Returns the album notes, image URLs, etc., using data from last.fm
|
|
|
|
aid:int The album ID
|
|
"""
|
|
methodName = 'getAlbumInfo'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'id': int(aid)}
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getAlbumInfo2(self, aid):
|
|
"""
|
|
since 1.14.0
|
|
|
|
Same as getAlbumInfo, but uses ID3 tags
|
|
|
|
aid:int The album ID
|
|
"""
|
|
methodName = 'getAlbumInfo2'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = {'id': int(aid)}
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def getCaptions(self, vid, fmt=None):
|
|
"""
|
|
since 1.14.0
|
|
|
|
Returns captions (subtitles) for a video. Use getVideoInfo for a list
|
|
of captions.
|
|
|
|
vid:int The ID of the video
|
|
fmt:str Preferred captions format ("srt" or "vtt")
|
|
"""
|
|
methodName = 'getCaptions'
|
|
viewName = '%s.view' % methodName
|
|
|
|
q = self._getQueryDict({'id': int(vid), 'format': fmt})
|
|
req = self._getRequest(viewName, q)
|
|
res = self._doInfoReq(req)
|
|
self._checkStatus(res)
|
|
return res
|
|
|
|
def _unsupportedAPIFunction(self, methodName):
|
|
"""
|
|
base function to call unsupported API methods
|
|
|
|
Returns True if refresh successful, False otherwise
|
|
:rtype : boolean
|
|
"""
|
|
baseMethod = 'musicFolderSettings'
|
|
viewName = '%s.view' % baseMethod
|
|
|
|
url = '%s:%d/%s/%s?%s' % (self._baseUrl, self._port,
|
|
self._separateServerPath(), viewName, methodName)
|
|
req = urllib2.Request(url)
|
|
res = self._opener.open(req)
|
|
res_msg = res.msg.lower()
|
|
return res_msg == 'ok'
|
|
|
|
#
|
|
# Private internal methods
|
|
#
|
|
def _getOpener(self, username, passwd):
|
|
# Context is only relevent in >= python 2.7.9
|
|
https_chain = HTTPSHandlerChain()
|
|
if sys.version_info[:3] >= (2, 7, 9) and self._insecure:
|
|
https_chain = HTTPSHandlerChain(
|
|
context=ssl._create_unverified_context())
|
|
opener = urllib2.build_opener(PysHTTPRedirectHandler, https_chain)
|
|
return opener
|
|
|
|
def _getQueryDict(self, d):
|
|
"""
|
|
Given a dictionary, it cleans out all the values set to None
|
|
"""
|
|
for k, v in d.items():
|
|
if v is None:
|
|
del d[k]
|
|
return d
|
|
|
|
def _getBaseQdict(self):
|
|
qdict = {
|
|
'f': 'json',
|
|
'v': self._apiVersion,
|
|
'c': self._appName,
|
|
'u': self._username,
|
|
}
|
|
|
|
if self._legacyAuth:
|
|
qdict['p'] = 'enc:%s' % self._hexEnc(self._rawPass)
|
|
else:
|
|
salt = self._getSalt()
|
|
token = md5(self._rawPass + salt).hexdigest()
|
|
qdict.update({
|
|
's': salt,
|
|
't': token,
|
|
})
|
|
|
|
return qdict
|
|
|
|
def _getRequest(self, viewName, query={}):
|
|
qdict = self._getBaseQdict()
|
|
qdict.update(query)
|
|
url = '%s:%d/%s/%s' % (self._baseUrl, self._port, self._serverPath,
|
|
viewName)
|
|
req = urllib2.Request(url, urlencode(qdict))
|
|
|
|
if self._useGET:
|
|
url += '?%s' % urlencode(qdict)
|
|
req = urllib2.Request(url)
|
|
|
|
return req
|
|
|
|
def _getRequestWithList(self, viewName, listName, alist, query={}):
|
|
"""
|
|
Like _getRequest, but allows appending a number of items with the
|
|
same key (listName). This bypasses the limitation of urlencode()
|
|
"""
|
|
qdict = self._getBaseQdict()
|
|
qdict.update(query)
|
|
url = '%s:%d/%s/%s' % (self._baseUrl, self._port, self._serverPath,
|
|
viewName)
|
|
data = StringIO()
|
|
data.write(urlencode(qdict))
|
|
for i in alist:
|
|
data.write('&%s' % urlencode({listName: i}))
|
|
req = urllib2.Request(url, data.getvalue())
|
|
|
|
if self._useGET:
|
|
url += '?%s' % data.getvalue()
|
|
req = urllib2.Request(url)
|
|
|
|
return req
|
|
|
|
def _getRequestWithLists(self, viewName, listMap, query={}):
|
|
"""
|
|
Like _getRequestWithList(), but you must pass a dictionary
|
|
that maps the listName to the list. This allows for multiple
|
|
list parameters to be used, like in updatePlaylist()
|
|
|
|
viewName:str The name of the view
|
|
listMap:dict A mapping of listName to a list of entries
|
|
query:dict The normal query dict
|
|
"""
|
|
qdict = self._getBaseQdict()
|
|
qdict.update(query)
|
|
url = '%s:%d/%s/%s' % (self._baseUrl, self._port, self._serverPath,
|
|
viewName)
|
|
data = StringIO()
|
|
data.write(urlencode(qdict))
|
|
for k, l in listMap.iteritems():
|
|
for i in l:
|
|
data.write('&%s' % urlencode({k: i}))
|
|
req = urllib2.Request(url, data.getvalue())
|
|
|
|
if self._useGET:
|
|
url += '?%s' % data.getvalue()
|
|
req = urllib2.Request(url)
|
|
|
|
return req
|
|
|
|
def _doInfoReq(self, req):
|
|
# Returns a parsed dictionary version of the result
|
|
res = self._opener.open(req)
|
|
dres = json.loads(res.read())
|
|
return dres['subsonic-response']
|
|
|
|
def _doBinReq(self, req):
|
|
res = self._opener.open(req)
|
|
contType = res.info().getheader('Content-Type')
|
|
if contType:
|
|
if contType.startswith('text/html') or \
|
|
contType.startswith('application/json'):
|
|
dres = json.loads(res.read())
|
|
return dres['subsonic-response']
|
|
return res
|
|
|
|
def _checkStatus(self, result):
|
|
if result['status'] == 'ok':
|
|
return True
|
|
elif result['status'] == 'failed':
|
|
exc = getExcByCode(result['error']['code'])
|
|
raise exc(result['error']['message'])
|
|
|
|
def _hexEnc(self, raw):
|
|
"""
|
|
Returns a "hex encoded" string per the Subsonic api docs
|
|
|
|
raw:str The string to hex encode
|
|
"""
|
|
ret = ''
|
|
for c in raw:
|
|
ret += '%02X' % ord(c)
|
|
return ret
|
|
|
|
def _ts2milli(self, ts):
|
|
"""
|
|
For whatever reason, Subsonic uses timestamps in milliseconds since
|
|
the unix epoch. I have no idea what need there is of this precision,
|
|
but this will just multiply the timestamp times 1000 and return the int
|
|
"""
|
|
if ts is None:
|
|
return None
|
|
return int(ts * 1000)
|
|
|
|
def _separateServerPath(self):
|
|
"""
|
|
separate REST portion of URL from base server path.
|
|
"""
|
|
return urllib2.splithost(self._serverPath)[1].split('/')[0]
|
|
|
|
def _fixLastModified(self, data):
|
|
"""
|
|
This will recursively walk through a data structure and look for
|
|
a dict key/value pair where the key is "lastModified" and change
|
|
the shitty java millisecond timestamp to a real unix timestamp
|
|
of SECONDS since the unix epoch. JAVA SUCKS!
|
|
"""
|
|
if isinstance(data, dict):
|
|
for k, v in data.items():
|
|
if k == 'lastModified':
|
|
data[k] = long(v) / 1000.0
|
|
return
|
|
elif isinstance(v, (tuple, list, dict)):
|
|
return self._fixLastModified(v)
|
|
elif isinstance(data, (list, tuple)):
|
|
for item in data:
|
|
if isinstance(item, (list, tuple, dict)):
|
|
return self._fixLastModified(item)
|
|
|
|
def _process_netrc(self, use_netrc):
|
|
"""
|
|
The use_netrc var is either a boolean, which means we should use
|
|
the user's default netrc, or a string specifying a path to a
|
|
netrc formatted file
|
|
|
|
use_netrc:bool|str Either set to True to use the user's default
|
|
netrc file or a string specifying a specific
|
|
netrc file to use
|
|
"""
|
|
if not use_netrc:
|
|
raise CredentialError('useNetrc must be either a boolean "True" '
|
|
'or a string representing a path to a netrc file, '
|
|
'not {0}'.format(repr(use_netrc)))
|
|
if isinstance(use_netrc, bool) and use_netrc:
|
|
self._netrc = netrc()
|
|
else:
|
|
# This should be a string specifying a path to a netrc file
|
|
self._netrc = netrc(os.path.expanduser(use_netrc))
|
|
auth = self._netrc.authenticators(self._hostname)
|
|
if not auth:
|
|
raise CredentialError('No machine entry found for {0} in '
|
|
'your netrc file'.format(self._hostname))
|
|
|
|
# If we get here, we have credentials
|
|
self._username = auth[0]
|
|
self._rawPass = auth[2]
|
|
|
|
def _getSalt(self, length=12):
|
|
salt = md5(os.urandom(100)).hexdigest()
|
|
return salt[:length]
|