events: initial implementation
This module use asyncio, so require Python 3. And actually >= 3.5.2 because of asyncio.StreamReader.readuntil(). Modules are designed the way it's still possible to use non-events API on Python 2.
This commit is contained in:
parent
d8b0ff349d
commit
785706af2f
@ -112,6 +112,8 @@ class QubesBase(qubesmgmt.base.PropertyHolder):
|
|||||||
labels = None
|
labels = None
|
||||||
#: storage pools
|
#: storage pools
|
||||||
pools = None
|
pools = None
|
||||||
|
#: type of qubesd connection: either 'socket' or 'qrexec'
|
||||||
|
qubesd_connection_type = None
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
super(QubesBase, self).__init__(self, 'mgmt.property.', 'dom0')
|
super(QubesBase, self).__init__(self, 'mgmt.property.', 'dom0')
|
||||||
@ -178,6 +180,9 @@ class QubesLocal(QubesBase):
|
|||||||
|
|
||||||
Used when running in dom0.
|
Used when running in dom0.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
|
qubesd_connection_type = 'socket'
|
||||||
|
|
||||||
def qubesd_call(self, dest, method, arg=None, payload=None):
|
def qubesd_call(self, dest, method, arg=None, payload=None):
|
||||||
try:
|
try:
|
||||||
client_socket = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
|
client_socket = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
|
||||||
@ -205,6 +210,9 @@ class QubesRemote(QubesBase):
|
|||||||
|
|
||||||
Used when running in VM.
|
Used when running in VM.
|
||||||
'''
|
'''
|
||||||
|
|
||||||
|
qubesd_connection_type = 'qrexec'
|
||||||
|
|
||||||
def qubesd_call(self, dest, method, arg=None, payload=None):
|
def qubesd_call(self, dest, method, arg=None, payload=None):
|
||||||
service_name = method
|
service_name = method
|
||||||
if arg is not None:
|
if arg is not None:
|
||||||
|
158
qubesmgmt/events/__init__.py
Normal file
158
qubesmgmt/events/__init__.py
Normal file
@ -0,0 +1,158 @@
|
|||||||
|
# -*- encoding: utf8 -*-
|
||||||
|
#
|
||||||
|
# The Qubes OS Project, http://www.qubes-os.org
|
||||||
|
#
|
||||||
|
# Copyright (C) 2017 Marek Marczykowski-Górecki
|
||||||
|
# <marmarek@invisiblethingslab.com>
|
||||||
|
#
|
||||||
|
# This program is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Lesser General Public License as published by
|
||||||
|
# the Free Software Foundation; either version 2 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Lesser General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Lesser General Public License along
|
||||||
|
# with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
'''Event handling implementation, require Python >=3.5.2 for asyncio.'''
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
import subprocess
|
||||||
|
|
||||||
|
import qubesmgmt.config
|
||||||
|
import qubesmgmt.exc
|
||||||
|
|
||||||
|
|
||||||
|
class EventsDispatcher(object):
|
||||||
|
''' Events dispatcher, responsible for receiving events and calling
|
||||||
|
appropriate handlers'''
|
||||||
|
def __init__(self, app):
|
||||||
|
'''Initialize EventsDispatcher'''
|
||||||
|
#: Qubes() object
|
||||||
|
self.app = app
|
||||||
|
|
||||||
|
#: event handlers - dict of event -> handlers
|
||||||
|
self.handlers = {}
|
||||||
|
|
||||||
|
def add_handler(self, event, handler):
|
||||||
|
'''Register handler for event
|
||||||
|
|
||||||
|
Use '*' as event to register a handler for all events.
|
||||||
|
|
||||||
|
Handler function is called with:
|
||||||
|
* subject (VM object or None)
|
||||||
|
* event name (str)
|
||||||
|
* keyword arguments related to the event, if any - all values as str
|
||||||
|
|
||||||
|
:param event Event name, or '*' for all events
|
||||||
|
:param handler Handler function'''
|
||||||
|
self.handlers.setdefault(event, set()).add(handler)
|
||||||
|
|
||||||
|
def remove_handler(self, event, handler):
|
||||||
|
'''Remove previously registered event handler
|
||||||
|
|
||||||
|
:param event Event name
|
||||||
|
:param handler Handler function
|
||||||
|
'''
|
||||||
|
self.handlers[event].remove(handler)
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def _get_events_reader(self, vm=None) -> (asyncio.StreamReader, callable):
|
||||||
|
'''Make connection to qubesd and return stream to read events from
|
||||||
|
|
||||||
|
:param vm: Specific VM for which events should be handled, use None
|
||||||
|
to handle events from all VMs (and non-VM objects)
|
||||||
|
:return stream to read events from and a cleanup function
|
||||||
|
(call it to terminate qubesd connection)'''
|
||||||
|
if vm is not None:
|
||||||
|
dest = vm.name
|
||||||
|
else:
|
||||||
|
dest = 'dom0'
|
||||||
|
|
||||||
|
if self.app.qubesd_connection_type == 'socket':
|
||||||
|
reader, writer = yield from asyncio.open_unix_connection(
|
||||||
|
qubesmgmt.config.QUBESD_SOCKET)
|
||||||
|
writer.write(b'dom0\0') # source
|
||||||
|
writer.write(b'mgmt.Events\0') # method
|
||||||
|
writer.write(dest.encode('ascii') + b'\0') # dest
|
||||||
|
writer.write(b'\0') # arg
|
||||||
|
writer.write_eof()
|
||||||
|
|
||||||
|
def cleanup_func():
|
||||||
|
'''Close connection to qubesd'''
|
||||||
|
writer.close()
|
||||||
|
elif self.app.qubesd_connection_type == 'qrexec':
|
||||||
|
proc = yield from asyncio.create_subprocess_exec(
|
||||||
|
['qrexec-client-vm', dest, 'mgmt.Events'],
|
||||||
|
stdin=subprocess.PIPE, stdout=subprocess.PIPE)
|
||||||
|
|
||||||
|
proc.stdin.write_eof()
|
||||||
|
reader = proc.stdout
|
||||||
|
|
||||||
|
def cleanup_func():
|
||||||
|
'''Close connection to qubesd'''
|
||||||
|
proc.kill()
|
||||||
|
else:
|
||||||
|
raise NotImplementedError('Unsupported qubesd connection type: '
|
||||||
|
+ self.app.qubesd_connection_type)
|
||||||
|
return reader, cleanup_func
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def listen_for_events(self, vm=None):
|
||||||
|
'''
|
||||||
|
Listen for events and call appropriate handlers.
|
||||||
|
This function do not exit until manually terminated.
|
||||||
|
|
||||||
|
This is coroutine.
|
||||||
|
|
||||||
|
:param vm: Listen for events only for this VM, use None to listen for
|
||||||
|
events about all VMs and not related to any particular VM.
|
||||||
|
:return: None
|
||||||
|
'''
|
||||||
|
|
||||||
|
try:
|
||||||
|
reader, cleanup_func = yield from self._get_events_reader(vm)
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
return
|
||||||
|
|
||||||
|
while not reader.at_eof():
|
||||||
|
try:
|
||||||
|
event_data = yield from reader.readuntil(b'\0\0')
|
||||||
|
if event_data == b'1\0\0':
|
||||||
|
# event with non-VM subject contains \0\0 inside of
|
||||||
|
# event, need to receive rest of the data
|
||||||
|
event_data += yield from reader.readuntil(b'\0\0')
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
break
|
||||||
|
except asyncio.IncompleteReadError as err:
|
||||||
|
if err.partial == b'':
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
raise
|
||||||
|
|
||||||
|
if not event_data.startswith(b'1\0'):
|
||||||
|
raise qubesmgmt.exc.QubesDaemonCommunicationError(
|
||||||
|
'Non-event received on events connection: '
|
||||||
|
+ repr(event_data))
|
||||||
|
event_data = event_data.decode('utf-8')
|
||||||
|
_, subject, event, *kwargs = event_data.split('\0')
|
||||||
|
# convert list to dict, remove last empty entry
|
||||||
|
kwargs = dict(zip(kwargs[:-2:2], kwargs[1:-2:2]))
|
||||||
|
self.handle(subject, event, **kwargs)
|
||||||
|
|
||||||
|
cleanup_func()
|
||||||
|
|
||||||
|
def handle(self, subject, event, **kwargs):
|
||||||
|
'''Call handlers for given event'''
|
||||||
|
if subject:
|
||||||
|
subject = self.app.domains[subject]
|
||||||
|
else:
|
||||||
|
subject = None
|
||||||
|
for handler in self.handlers.get(event, []):
|
||||||
|
handler(subject, event, **kwargs)
|
||||||
|
for handler in self.handlers.get('*', []):
|
||||||
|
handler(subject, event, **kwargs)
|
Loading…
Reference in New Issue
Block a user