]> kaliko git repositories - python-musicpd.git/commitdiff
adding command_list support
authorJ. Alexander Treuman <jat@spatialrift.net>
Sun, 24 Feb 2008 22:25:16 +0000 (17:25 -0500)
committerJ. Alexander Treuman <jat@spatialrift.net>
Sun, 24 Feb 2008 22:25:16 +0000 (17:25 -0500)
mpd.py

diff --git a/mpd.py b/mpd.py
index 7e92cc084426290971b0b00d37ca543a35befe60..2fdce12620d406e9f90597a3118d617cac270015 100644 (file)
--- a/mpd.py
+++ b/mpd.py
@@ -3,7 +3,6 @@
 # TODO: return {} if no object read (?)
 # TODO: implement argument checking/parsing (?)
 # TODO: check for EOF when reading and benchmark it
-# TODO: command_list support
 # TODO: converter support
 # TODO: global for parsing MPD_HOST/MPD_PORT
 # TODO: global for parsing MPD error messages
@@ -15,6 +14,7 @@ import socket
 HELLO_PREFIX = "OK MPD "
 ERROR_PREFIX = "ACK "
 SUCCESS = "OK"
+NEXT = "list_OK"
 
 
 class MPDError(Exception):
@@ -26,6 +26,9 @@ class ProtocolError(MPDError):
 class CommandError(MPDError):
     pass
 
+class CommandListError(MPDError):
+    pass
+
 
 class MPDClient(object):
     def __init__(self):
@@ -112,16 +115,20 @@ class MPDClient(object):
         return lambda *args: self._docommand(attr, args, retval)
 
     def _docommand(self, command, args, retval):
+        if self._commandlist is not None and not callable(retval):
+            raise CommandListError, "%s not allowed in command list" % command
         self._writecommand(command, args)
-        if callable(retval):
-            return retval()
-        return retval
+        if self._commandlist is None:
+            if callable(retval):
+                return retval()
+            return retval
+        self._commandlist.append(retval)
 
     def _writeline(self, line):
         self._sockfile.write("%s\n" % line)
         self._sockfile.flush()
 
-    def _writecommand(self, command, args):
+    def _writecommand(self, command, args=[]):
         parts = [command]
         for arg in args:
             parts.append('"%s"' % escape(str(arg)))
@@ -132,7 +139,12 @@ class MPDClient(object):
         if line.startswith(ERROR_PREFIX):
             error = line[len(ERROR_PREFIX):].strip()
             raise CommandError, error
-        if line == SUCCESS:
+        if self._commandlist is not None:
+            if line == NEXT:
+                return
+            if line == SUCCESS:
+                raise ProtocolError, "Got unexpected '%s'" % SUCCESS
+        elif line == SUCCESS:
             return
         return line
 
@@ -187,6 +199,13 @@ class MPDClient(object):
             yield obj
         raise StopIteration
 
+    def _readcommandlist(self):
+        for retval in self._commandlist:
+            yield retval()
+        self._commandlist = None
+        self._getnone()
+        raise StopIteration
+
     def _wrapiterator(self, iterator):
         if not self.iterate:
             return list(iterator)
@@ -230,6 +249,9 @@ class MPDClient(object):
     def _getchanges(self):
         return self._getobjects(["cpos"])
 
+    def _getcommandlist(self):
+        return self._wrapiterator(self._readcommandlist())
+
     def _hello(self):
         line = self._sockfile.readline().rstrip("\n")
         if not line.startswith(HELLO_PREFIX):
@@ -238,6 +260,7 @@ class MPDClient(object):
 
     def _reset(self):
         self.mpd_version = None
+        self._commandlist = None
         self._sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
         self._sockfile = self._sock.makefile("rb+")
 
@@ -251,6 +274,18 @@ class MPDClient(object):
         self._sock.close()
         self._reset()
 
+    def command_list_ok_begin(self):
+        if self._commandlist is not None:
+            raise CommandListError, "Already in command list"
+        self._writecommand("command_list_ok_begin")
+        self._commandlist = []
+
+    def command_list_end(self):
+        if self._commandlist is None:
+            raise CommandListError, "Not in command list"
+        self._writecommand("command_list_end")
+        return self._getcommandlist()
+
 
 def escape(text):
     return text.replace("\\", "\\\\").replace('"', '\\"')