blob: c7dbea076da6a6afe16dc71d6293489302e9defe [file] [log] [blame]
Luiz Capitulinocedebda2009-11-26 22:59:09 -02001# QEMU Monitor Protocol Python class
2#
Luiz Capitulino1d00a072010-10-27 17:43:34 -02003# Copyright (C) 2009, 2010 Red Hat Inc.
Luiz Capitulinocedebda2009-11-26 22:59:09 -02004#
5# Authors:
6# Luiz Capitulino <lcapitulino@redhat.com>
7#
8# This work is licensed under the terms of the GNU GPL, version 2. See
9# the COPYING file in the top-level directory.
10
Luiz Capitulino1d00a072010-10-27 17:43:34 -020011import json
12import errno
13import socket
Luiz Capitulinocedebda2009-11-26 22:59:09 -020014
15class QMPError(Exception):
16 pass
17
18class QMPConnectError(QMPError):
19 pass
20
Luiz Capitulino1d00a072010-10-27 17:43:34 -020021class QMPCapabilitiesError(QMPError):
22 pass
23
Luiz Capitulinocedebda2009-11-26 22:59:09 -020024class QEMUMonitorProtocol:
Stefan Hajnoczi37628f12011-05-25 19:48:01 +010025 def __init__(self, address, server=False):
Luiz Capitulino1d00a072010-10-27 17:43:34 -020026 """
27 Create a QEMUMonitorProtocol class.
Luiz Capitulinocedebda2009-11-26 22:59:09 -020028
Luiz Capitulino1d00a072010-10-27 17:43:34 -020029 @param address: QEMU address, can be either a unix socket path (string)
30 or a tuple in the form ( address, port ) for a TCP
31 connection
Stefan Hajnoczi37628f12011-05-25 19:48:01 +010032 @param server: server mode listens on the socket (bool)
33 @raise socket.error on socket connection errors
34 @note No connection is established, this is done by the connect() or
35 accept() methods
Luiz Capitulino1d00a072010-10-27 17:43:34 -020036 """
37 self.__events = []
38 self.__address = address
39 self.__sock = self.__get_sock()
Stefan Hajnoczi37628f12011-05-25 19:48:01 +010040 if server:
41 self.__sock.bind(self.__address)
42 self.__sock.listen(1)
Luiz Capitulinocedebda2009-11-26 22:59:09 -020043
Luiz Capitulino1d00a072010-10-27 17:43:34 -020044 def __get_sock(self):
45 if isinstance(self.__address, tuple):
46 family = socket.AF_INET
Luiz Capitulinocedebda2009-11-26 22:59:09 -020047 else:
Luiz Capitulino1d00a072010-10-27 17:43:34 -020048 family = socket.AF_UNIX
49 return socket.socket(family, socket.SOCK_STREAM)
Luiz Capitulinocedebda2009-11-26 22:59:09 -020050
Stefan Hajnoczi37628f12011-05-25 19:48:01 +010051 def __negotiate_capabilities(self):
52 self.__sockfile = self.__sock.makefile()
53 greeting = self.__json_read()
54 if greeting is None or not greeting.has_key('QMP'):
55 raise QMPConnectError
56 # Greeting seems ok, negotiate capabilities
57 resp = self.cmd('qmp_capabilities')
58 if "return" in resp:
59 return greeting
60 raise QMPCapabilitiesError
61
Stefan Hajnoczi91b8edd2011-05-25 19:48:00 +010062 def __json_read(self, only_event=False):
Luiz Capitulino1d00a072010-10-27 17:43:34 -020063 while True:
64 data = self.__sockfile.readline()
65 if not data:
66 return
67 resp = json.loads(data)
68 if 'event' in resp:
69 self.__events.append(resp)
Stefan Hajnoczi91b8edd2011-05-25 19:48:00 +010070 if not only_event:
71 continue
Luiz Capitulino1d00a072010-10-27 17:43:34 -020072 return resp
Luiz Capitulinocedebda2009-11-26 22:59:09 -020073
Luiz Capitulino1d00a072010-10-27 17:43:34 -020074 error = socket.error
75
76 def connect(self):
77 """
78 Connect to the QMP Monitor and perform capabilities negotiation.
79
80 @return QMP greeting dict
81 @raise socket.error on socket connection errors
82 @raise QMPConnectError if the greeting is not received
83 @raise QMPCapabilitiesError if fails to negotiate capabilities
84 """
85 self.__sock.connect(self.__address)
Stefan Hajnoczi37628f12011-05-25 19:48:01 +010086 return self.__negotiate_capabilities()
87
88 def accept(self):
89 """
90 Await connection from QMP Monitor and perform capabilities negotiation.
91
92 @return QMP greeting dict
93 @raise socket.error on socket connection errors
94 @raise QMPConnectError if the greeting is not received
95 @raise QMPCapabilitiesError if fails to negotiate capabilities
96 """
97 self.__sock, _ = self.__sock.accept()
98 return self.__negotiate_capabilities()
Luiz Capitulino1d00a072010-10-27 17:43:34 -020099
100 def cmd_obj(self, qmp_cmd):
101 """
102 Send a QMP command to the QMP Monitor.
103
104 @param qmp_cmd: QMP command to be sent as a Python dict
105 @return QMP response as a Python dict or None if the connection has
106 been closed
107 """
108 try:
109 self.__sock.sendall(json.dumps(qmp_cmd))
110 except socket.error, err:
111 if err[0] == errno.EPIPE:
112 return
113 raise socket.error(err)
114 return self.__json_read()
115
116 def cmd(self, name, args=None, id=None):
117 """
118 Build a QMP command and send it to the QMP Monitor.
119
120 @param name: command name (string)
121 @param args: command arguments (dict)
122 @param id: command id (dict, list, string or int)
123 """
124 qmp_cmd = { 'execute': name }
125 if args:
126 qmp_cmd['arguments'] = args
127 if id:
128 qmp_cmd['id'] = id
129 return self.cmd_obj(qmp_cmd)
130
Stefan Hajnoczi91b8edd2011-05-25 19:48:00 +0100131 def get_events(self, wait=False):
Luiz Capitulino1d00a072010-10-27 17:43:34 -0200132 """
133 Get a list of available QMP events.
Stefan Hajnoczi91b8edd2011-05-25 19:48:00 +0100134
135 @param wait: block until an event is available (bool)
Luiz Capitulino1d00a072010-10-27 17:43:34 -0200136 """
137 self.__sock.setblocking(0)
138 try:
139 self.__json_read()
140 except socket.error, err:
141 if err[0] == errno.EAGAIN:
142 # No data available
143 pass
144 self.__sock.setblocking(1)
Stefan Hajnoczi91b8edd2011-05-25 19:48:00 +0100145 if not self.__events and wait:
146 self.__json_read(only_event=True)
Luiz Capitulino1d00a072010-10-27 17:43:34 -0200147 return self.__events
148
149 def clear_events(self):
150 """
151 Clear current list of pending events.
152 """
153 self.__events = []
154
155 def close(self):
156 self.__sock.close()
157 self.__sockfile.close()