1341 lines
43 KiB
Diff
1341 lines
43 KiB
Diff
|
From 1f753894b1a5a3f17d1452a572a9a126fa526998 Mon Sep 17 00:00:00 2001
|
||
|
From: Jochen Breuer <jbreuer@suse.de>
|
||
|
Date: Fri, 3 Jul 2020 14:43:53 +0200
|
||
|
Subject: [PATCH] Opensuse 3000 libvirt engine fixes (#251)
|
||
|
MIME-Version: 1.0
|
||
|
Content-Type: text/plain; charset=UTF-8
|
||
|
Content-Transfer-Encoding: 8bit
|
||
|
|
||
|
* Blacken libvirt-engines
|
||
|
|
||
|
* libvirt events engine: constant fixes
|
||
|
|
||
|
The libvirt constants for pool and nodedev events need tweaking since
|
||
|
they don't match the rule used to autogenerate them. Fixes these warnings
|
||
|
at the engine start.
|
||
|
|
||
|
Skipping "pool/lifecycle" events: libvirt too old
|
||
|
Skipping "pool/refresh" events: libvirt too old
|
||
|
Skipping "nodedev/lifecycle" events: libvirt too old
|
||
|
Skipping "nodedev/update" events: libvirt too old
|
||
|
|
||
|
Co-authored-by: Cédric Bosdonnat <cbosdonnat@suse.com>
|
||
|
---
|
||
|
changelog/57746.fixed | 1 +
|
||
|
salt/engines/libvirt_events.py | 685 ++++++++++++----------
|
||
|
tests/unit/engines/test_libvirt_events.py | 187 +++---
|
||
|
3 files changed, 495 insertions(+), 378 deletions(-)
|
||
|
create mode 100644 changelog/57746.fixed
|
||
|
|
||
|
diff --git a/changelog/57746.fixed b/changelog/57746.fixed
|
||
|
new file mode 100644
|
||
|
index 0000000000..5102bb04e9
|
||
|
--- /dev/null
|
||
|
+++ b/changelog/57746.fixed
|
||
|
@@ -0,0 +1 @@
|
||
|
+Fix the registration of libvirt pool and nodedev events
|
||
|
diff --git a/salt/engines/libvirt_events.py b/salt/engines/libvirt_events.py
|
||
|
index cdb5d1dfe8..96ba9efc91 100644
|
||
|
--- a/salt/engines/libvirt_events.py
|
||
|
+++ b/salt/engines/libvirt_events.py
|
||
|
@@ -1,6 +1,6 @@
|
||
|
# -*- coding: utf-8 -*-
|
||
|
|
||
|
-'''
|
||
|
+"""
|
||
|
An engine that listens for libvirt events and resends them to the salt event bus.
|
||
|
|
||
|
The minimal configuration is the following and will listen to all events on the
|
||
|
@@ -35,7 +35,7 @@ CALLBACK_DEFS constant. If the filters list contains ``all``, all
|
||
|
events will be relayed.
|
||
|
|
||
|
Be aware that the list of events increases with libvirt versions, for example
|
||
|
-network events have been added in libvirt 1.2.1.
|
||
|
+network events have been added in libvirt 1.2.1 and storage events in 2.0.0.
|
||
|
|
||
|
Running the engine on non-root
|
||
|
------------------------------
|
||
|
@@ -63,7 +63,7 @@ A polkit rule like the following one will allow `salt` user to connect to libvir
|
||
|
:depends: libvirt 1.0.0+ python binding
|
||
|
|
||
|
.. versionadded:: 2019.2.0
|
||
|
-'''
|
||
|
+"""
|
||
|
|
||
|
from __future__ import absolute_import, unicode_literals, print_function
|
||
|
import logging
|
||
|
@@ -73,6 +73,7 @@ import salt.utils.event
|
||
|
|
||
|
# pylint: disable=no-name-in-module,import-error
|
||
|
from salt.ext.six.moves.urllib.parse import urlparse
|
||
|
+
|
||
|
# pylint: enable=no-name-in-module,import-error
|
||
|
|
||
|
log = logging.getLogger(__name__)
|
||
|
@@ -85,112 +86,125 @@ except ImportError:
|
||
|
|
||
|
|
||
|
def __virtual__():
|
||
|
- '''
|
||
|
+ """
|
||
|
Only load if libvirt python binding is present
|
||
|
- '''
|
||
|
+ """
|
||
|
if libvirt is None:
|
||
|
- msg = 'libvirt module not found'
|
||
|
+ msg = "libvirt module not found"
|
||
|
elif libvirt.getVersion() < 1000000:
|
||
|
- msg = 'libvirt >= 1.0.0 required'
|
||
|
+ msg = "libvirt >= 1.0.0 required"
|
||
|
else:
|
||
|
- msg = ''
|
||
|
+ msg = ""
|
||
|
return not bool(msg), msg
|
||
|
|
||
|
|
||
|
REGISTER_FUNCTIONS = {
|
||
|
- 'domain': 'domainEventRegisterAny',
|
||
|
- 'network': 'networkEventRegisterAny',
|
||
|
- 'pool': 'storagePoolEventRegisterAny',
|
||
|
- 'nodedev': 'nodeDeviceEventRegisterAny',
|
||
|
- 'secret': 'secretEventRegisterAny'
|
||
|
+ "domain": "domainEventRegisterAny",
|
||
|
+ "network": "networkEventRegisterAny",
|
||
|
+ "pool": "storagePoolEventRegisterAny",
|
||
|
+ "nodedev": "nodeDeviceEventRegisterAny",
|
||
|
+ "secret": "secretEventRegisterAny",
|
||
|
}
|
||
|
|
||
|
# Handle either BLOCK_JOB or BLOCK_JOB_2, but prefer the latter
|
||
|
-if hasattr(libvirt, 'VIR_DOMAIN_EVENT_ID_BLOCK_JOB_2'):
|
||
|
- BLOCK_JOB_ID = 'VIR_DOMAIN_EVENT_ID_BLOCK_JOB_2'
|
||
|
+if hasattr(libvirt, "VIR_DOMAIN_EVENT_ID_BLOCK_JOB_2"):
|
||
|
+ BLOCK_JOB_ID = "VIR_DOMAIN_EVENT_ID_BLOCK_JOB_2"
|
||
|
else:
|
||
|
- BLOCK_JOB_ID = 'VIR_DOMAIN_EVENT_ID_BLOCK_JOB'
|
||
|
+ BLOCK_JOB_ID = "VIR_DOMAIN_EVENT_ID_BLOCK_JOB"
|
||
|
|
||
|
CALLBACK_DEFS = {
|
||
|
- 'domain': (('lifecycle', None),
|
||
|
- ('reboot', None),
|
||
|
- ('rtc_change', None),
|
||
|
- ('watchdog', None),
|
||
|
- ('graphics', None),
|
||
|
- ('io_error', 'VIR_DOMAIN_EVENT_ID_IO_ERROR_REASON'),
|
||
|
- ('control_error', None),
|
||
|
- ('disk_change', None),
|
||
|
- ('tray_change', None),
|
||
|
- ('pmwakeup', None),
|
||
|
- ('pmsuspend', None),
|
||
|
- ('balloon_change', None),
|
||
|
- ('pmsuspend_disk', None),
|
||
|
- ('device_removed', None),
|
||
|
- ('block_job', BLOCK_JOB_ID),
|
||
|
- ('tunable', None),
|
||
|
- ('agent_lifecycle', None),
|
||
|
- ('device_added', None),
|
||
|
- ('migration_iteration', None),
|
||
|
- ('job_completed', None),
|
||
|
- ('device_removal_failed', None),
|
||
|
- ('metadata_change', None),
|
||
|
- ('block_threshold', None)),
|
||
|
- 'network': (('lifecycle', None),),
|
||
|
- 'pool': (('lifecycle', None),
|
||
|
- ('refresh', None)),
|
||
|
- 'nodedev': (('lifecycle', None),
|
||
|
- ('update', None)),
|
||
|
- 'secret': (('lifecycle', None),
|
||
|
- ('value_changed', None))
|
||
|
+ "domain": (
|
||
|
+ ("lifecycle", None),
|
||
|
+ ("reboot", None),
|
||
|
+ ("rtc_change", None),
|
||
|
+ ("watchdog", None),
|
||
|
+ ("graphics", None),
|
||
|
+ ("io_error", "VIR_DOMAIN_EVENT_ID_IO_ERROR_REASON"),
|
||
|
+ ("control_error", None),
|
||
|
+ ("disk_change", None),
|
||
|
+ ("tray_change", None),
|
||
|
+ ("pmwakeup", None),
|
||
|
+ ("pmsuspend", None),
|
||
|
+ ("balloon_change", None),
|
||
|
+ ("pmsuspend_disk", None),
|
||
|
+ ("device_removed", None),
|
||
|
+ ("block_job", BLOCK_JOB_ID),
|
||
|
+ ("tunable", None),
|
||
|
+ ("agent_lifecycle", None),
|
||
|
+ ("device_added", None),
|
||
|
+ ("migration_iteration", None),
|
||
|
+ ("job_completed", None),
|
||
|
+ ("device_removal_failed", None),
|
||
|
+ ("metadata_change", None),
|
||
|
+ ("block_threshold", None),
|
||
|
+ ),
|
||
|
+ "network": (("lifecycle", None),),
|
||
|
+ "pool": (
|
||
|
+ ("lifecycle", "VIR_STORAGE_POOL_EVENT_ID_LIFECYCLE"),
|
||
|
+ ("refresh", "VIR_STORAGE_POOL_EVENT_ID_REFRESH"),
|
||
|
+ ),
|
||
|
+ "nodedev": (
|
||
|
+ ("lifecycle", "VIR_NODE_DEVICE_EVENT_ID_LIFECYCLE"),
|
||
|
+ ("update", "VIR_NODE_DEVICE_EVENT_ID_UPDATE"),
|
||
|
+ ),
|
||
|
+ "secret": (("lifecycle", None), ("value_changed", None)),
|
||
|
}
|
||
|
|
||
|
|
||
|
def _compute_subprefix(attr):
|
||
|
- '''
|
||
|
+ """
|
||
|
Get the part before the first '_' or the end of attr including
|
||
|
the potential '_'
|
||
|
- '''
|
||
|
- return ''.join((attr.split('_')[0], '_' if len(attr.split('_')) > 1 else ''))
|
||
|
+ """
|
||
|
+ return "".join((attr.split("_")[0], "_" if len(attr.split("_")) > 1 else ""))
|
||
|
|
||
|
|
||
|
def _get_libvirt_enum_string(prefix, value):
|
||
|
- '''
|
||
|
+ """
|
||
|
Convert the libvirt enum integer value into a human readable string.
|
||
|
|
||
|
:param prefix: start of the libvirt attribute to look for.
|
||
|
:param value: integer to convert to string
|
||
|
- '''
|
||
|
- attributes = [attr[len(prefix):] for attr in libvirt.__dict__ if attr.startswith(prefix)]
|
||
|
+ """
|
||
|
+ attributes = [
|
||
|
+ attr[len(prefix) :] for attr in libvirt.__dict__ if attr.startswith(prefix)
|
||
|
+ ]
|
||
|
|
||
|
# Filter out the values starting with a common base as they match another enum
|
||
|
prefixes = [_compute_subprefix(p) for p in attributes]
|
||
|
counts = {p: prefixes.count(p) for p in prefixes}
|
||
|
- sub_prefixes = [p for p, count in counts.items() if count > 1 or (p.endswith('_') and p[:-1] in prefixes)]
|
||
|
- filtered = [attr for attr in attributes if _compute_subprefix(attr) not in sub_prefixes]
|
||
|
+ sub_prefixes = [
|
||
|
+ p
|
||
|
+ for p, count in counts.items()
|
||
|
+ if count > 1 or (p.endswith("_") and p[:-1] in prefixes)
|
||
|
+ ]
|
||
|
+ filtered = [
|
||
|
+ attr for attr in attributes if _compute_subprefix(attr) not in sub_prefixes
|
||
|
+ ]
|
||
|
|
||
|
for candidate in filtered:
|
||
|
- if value == getattr(libvirt, ''.join((prefix, candidate))):
|
||
|
- name = candidate.lower().replace('_', ' ')
|
||
|
+ if value == getattr(libvirt, "".join((prefix, candidate))):
|
||
|
+ name = candidate.lower().replace("_", " ")
|
||
|
return name
|
||
|
- return 'unknown'
|
||
|
+ return "unknown"
|
||
|
|
||
|
|
||
|
def _get_domain_event_detail(event, detail):
|
||
|
- '''
|
||
|
+ """
|
||
|
Convert event and detail numeric values into a tuple of human readable strings
|
||
|
- '''
|
||
|
- event_name = _get_libvirt_enum_string('VIR_DOMAIN_EVENT_', event)
|
||
|
- if event_name == 'unknown':
|
||
|
- return event_name, 'unknown'
|
||
|
+ """
|
||
|
+ event_name = _get_libvirt_enum_string("VIR_DOMAIN_EVENT_", event)
|
||
|
+ if event_name == "unknown":
|
||
|
+ return event_name, "unknown"
|
||
|
|
||
|
- prefix = 'VIR_DOMAIN_EVENT_{0}_'.format(event_name.upper())
|
||
|
+ prefix = "VIR_DOMAIN_EVENT_{0}_".format(event_name.upper())
|
||
|
detail_name = _get_libvirt_enum_string(prefix, detail)
|
||
|
|
||
|
return event_name, detail_name
|
||
|
|
||
|
|
||
|
def _salt_send_event(opaque, conn, data):
|
||
|
- '''
|
||
|
+ """
|
||
|
Convenience function adding common data to the event and sending it
|
||
|
on the salt event bus.
|
||
|
|
||
|
@@ -198,10 +212,10 @@ def _salt_send_event(opaque, conn, data):
|
||
|
This is a dict with 'prefix', 'object' and 'event' keys.
|
||
|
:param conn: libvirt connection
|
||
|
:param data: additional event data dict to send
|
||
|
- '''
|
||
|
- tag_prefix = opaque['prefix']
|
||
|
- object_type = opaque['object']
|
||
|
- event_type = opaque['event']
|
||
|
+ """
|
||
|
+ tag_prefix = opaque["prefix"]
|
||
|
+ object_type = opaque["object"]
|
||
|
+ event_type = opaque["event"]
|
||
|
|
||
|
# Prepare the connection URI to fit in the tag
|
||
|
# qemu+ssh://user@host:1234/system -> qemu+ssh/user@host:1234/system
|
||
|
@@ -209,30 +223,28 @@ def _salt_send_event(opaque, conn, data):
|
||
|
uri_tag = [uri.scheme]
|
||
|
if uri.netloc:
|
||
|
uri_tag.append(uri.netloc)
|
||
|
- path = uri.path.strip('/')
|
||
|
+ path = uri.path.strip("/")
|
||
|
if path:
|
||
|
uri_tag.append(path)
|
||
|
uri_str = "/".join(uri_tag)
|
||
|
|
||
|
# Append some common data
|
||
|
- all_data = {
|
||
|
- 'uri': conn.getURI()
|
||
|
- }
|
||
|
+ all_data = {"uri": conn.getURI()}
|
||
|
all_data.update(data)
|
||
|
|
||
|
- tag = '/'.join((tag_prefix, uri_str, object_type, event_type))
|
||
|
+ tag = "/".join((tag_prefix, uri_str, object_type, event_type))
|
||
|
|
||
|
# Actually send the event in salt
|
||
|
- if __opts__.get('__role') == 'master':
|
||
|
- salt.utils.event.get_master_event(
|
||
|
- __opts__,
|
||
|
- __opts__['sock_dir']).fire_event(all_data, tag)
|
||
|
+ if __opts__.get("__role") == "master":
|
||
|
+ salt.utils.event.get_master_event(__opts__, __opts__["sock_dir"]).fire_event(
|
||
|
+ all_data, tag
|
||
|
+ )
|
||
|
else:
|
||
|
- __salt__['event.send'](tag, all_data)
|
||
|
+ __salt__["event.send"](tag, all_data)
|
||
|
|
||
|
|
||
|
def _salt_send_domain_event(opaque, conn, domain, event, event_data):
|
||
|
- '''
|
||
|
+ """
|
||
|
Helper function send a salt event for a libvirt domain.
|
||
|
|
||
|
:param opaque: the opaque data that is passed to the callback.
|
||
|
@@ -241,375 +253,428 @@ def _salt_send_domain_event(opaque, conn, domain, event, event_data):
|
||
|
:param domain: name of the domain related to the event
|
||
|
:param event: name of the event
|
||
|
:param event_data: additional event data dict to send
|
||
|
- '''
|
||
|
+ """
|
||
|
data = {
|
||
|
- 'domain': {
|
||
|
- 'name': domain.name(),
|
||
|
- 'id': domain.ID(),
|
||
|
- 'uuid': domain.UUIDString()
|
||
|
+ "domain": {
|
||
|
+ "name": domain.name(),
|
||
|
+ "id": domain.ID(),
|
||
|
+ "uuid": domain.UUIDString(),
|
||
|
},
|
||
|
- 'event': event
|
||
|
+ "event": event,
|
||
|
}
|
||
|
data.update(event_data)
|
||
|
_salt_send_event(opaque, conn, data)
|
||
|
|
||
|
|
||
|
def _domain_event_lifecycle_cb(conn, domain, event, detail, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain lifecycle events handler
|
||
|
- '''
|
||
|
+ """
|
||
|
event_str, detail_str = _get_domain_event_detail(event, detail)
|
||
|
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'event': event_str,
|
||
|
- 'detail': detail_str
|
||
|
- })
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ domain,
|
||
|
+ opaque["event"],
|
||
|
+ {"event": event_str, "detail": detail_str},
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_reboot_cb(conn, domain, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain reboot events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {})
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(opaque, conn, domain, opaque["event"], {})
|
||
|
|
||
|
|
||
|
def _domain_event_rtc_change_cb(conn, domain, utcoffset, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain RTC change events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'utcoffset': utcoffset
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque, conn, domain, opaque["event"], {"utcoffset": utcoffset}
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_watchdog_cb(conn, domain, action, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain watchdog events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'action': _get_libvirt_enum_string('VIR_DOMAIN_EVENT_WATCHDOG_', action)
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ domain,
|
||
|
+ opaque["event"],
|
||
|
+ {"action": _get_libvirt_enum_string("VIR_DOMAIN_EVENT_WATCHDOG_", action)},
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_io_error_cb(conn, domain, srcpath, devalias, action, reason, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain I/O Error events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'srcPath': srcpath,
|
||
|
- 'dev': devalias,
|
||
|
- 'action': _get_libvirt_enum_string('VIR_DOMAIN_EVENT_IO_ERROR_', action),
|
||
|
- 'reason': reason
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ domain,
|
||
|
+ opaque["event"],
|
||
|
+ {
|
||
|
+ "srcPath": srcpath,
|
||
|
+ "dev": devalias,
|
||
|
+ "action": _get_libvirt_enum_string("VIR_DOMAIN_EVENT_IO_ERROR_", action),
|
||
|
+ "reason": reason,
|
||
|
+ },
|
||
|
+ )
|
||
|
|
||
|
|
||
|
-def _domain_event_graphics_cb(conn, domain, phase, local, remote, auth, subject, opaque):
|
||
|
- '''
|
||
|
+def _domain_event_graphics_cb(
|
||
|
+ conn, domain, phase, local, remote, auth, subject, opaque
|
||
|
+):
|
||
|
+ """
|
||
|
Domain graphics events handler
|
||
|
- '''
|
||
|
- prefix = 'VIR_DOMAIN_EVENT_GRAPHICS_'
|
||
|
+ """
|
||
|
+ prefix = "VIR_DOMAIN_EVENT_GRAPHICS_"
|
||
|
|
||
|
def get_address(addr):
|
||
|
- '''
|
||
|
+ """
|
||
|
transform address structure into event data piece
|
||
|
- '''
|
||
|
- return {'family': _get_libvirt_enum_string('{0}_ADDRESS_'.format(prefix), addr['family']),
|
||
|
- 'node': addr['node'],
|
||
|
- 'service': addr['service']}
|
||
|
+ """
|
||
|
+ return {
|
||
|
+ "family": _get_libvirt_enum_string(
|
||
|
+ "{0}_ADDRESS_".format(prefix), addr["family"]
|
||
|
+ ),
|
||
|
+ "node": addr["node"],
|
||
|
+ "service": addr["service"],
|
||
|
+ }
|
||
|
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'phase': _get_libvirt_enum_string(prefix, phase),
|
||
|
- 'local': get_address(local),
|
||
|
- 'remote': get_address(remote),
|
||
|
- 'authScheme': auth,
|
||
|
- 'subject': [{'type': item[0], 'name': item[1]} for item in subject]
|
||
|
- })
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ domain,
|
||
|
+ opaque["event"],
|
||
|
+ {
|
||
|
+ "phase": _get_libvirt_enum_string(prefix, phase),
|
||
|
+ "local": get_address(local),
|
||
|
+ "remote": get_address(remote),
|
||
|
+ "authScheme": auth,
|
||
|
+ "subject": [{"type": item[0], "name": item[1]} for item in subject],
|
||
|
+ },
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_control_error_cb(conn, domain, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain control error events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {})
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(opaque, conn, domain, opaque["event"], {})
|
||
|
|
||
|
|
||
|
def _domain_event_disk_change_cb(conn, domain, old_src, new_src, dev, reason, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain disk change events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'oldSrcPath': old_src,
|
||
|
- 'newSrcPath': new_src,
|
||
|
- 'dev': dev,
|
||
|
- 'reason': _get_libvirt_enum_string('VIR_DOMAIN_EVENT_DISK_', reason)
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ domain,
|
||
|
+ opaque["event"],
|
||
|
+ {
|
||
|
+ "oldSrcPath": old_src,
|
||
|
+ "newSrcPath": new_src,
|
||
|
+ "dev": dev,
|
||
|
+ "reason": _get_libvirt_enum_string("VIR_DOMAIN_EVENT_DISK_", reason),
|
||
|
+ },
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_tray_change_cb(conn, domain, dev, reason, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain tray change events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'dev': dev,
|
||
|
- 'reason': _get_libvirt_enum_string('VIR_DOMAIN_EVENT_TRAY_CHANGE_', reason)
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ domain,
|
||
|
+ opaque["event"],
|
||
|
+ {
|
||
|
+ "dev": dev,
|
||
|
+ "reason": _get_libvirt_enum_string("VIR_DOMAIN_EVENT_TRAY_CHANGE_", reason),
|
||
|
+ },
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_pmwakeup_cb(conn, domain, reason, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain wakeup events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'reason': 'unknown' # currently unused
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque, conn, domain, opaque["event"], {"reason": "unknown"} # currently unused
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_pmsuspend_cb(conn, domain, reason, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain suspend events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'reason': 'unknown' # currently unused
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque, conn, domain, opaque["event"], {"reason": "unknown"} # currently unused
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_balloon_change_cb(conn, domain, actual, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain balloon change events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'actual': actual
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(opaque, conn, domain, opaque["event"], {"actual": actual})
|
||
|
|
||
|
|
||
|
def _domain_event_pmsuspend_disk_cb(conn, domain, reason, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain disk suspend events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'reason': 'unknown' # currently unused
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque, conn, domain, opaque["event"], {"reason": "unknown"} # currently unused
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_block_job_cb(conn, domain, disk, job_type, status, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain block job events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'disk': disk,
|
||
|
- 'type': _get_libvirt_enum_string('VIR_DOMAIN_BLOCK_JOB_TYPE_', job_type),
|
||
|
- 'status': _get_libvirt_enum_string('VIR_DOMAIN_BLOCK_JOB_', status)
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ domain,
|
||
|
+ opaque["event"],
|
||
|
+ {
|
||
|
+ "disk": disk,
|
||
|
+ "type": _get_libvirt_enum_string("VIR_DOMAIN_BLOCK_JOB_TYPE_", job_type),
|
||
|
+ "status": _get_libvirt_enum_string("VIR_DOMAIN_BLOCK_JOB_", status),
|
||
|
+ },
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_device_removed_cb(conn, domain, dev, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain device removal events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'dev': dev
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(opaque, conn, domain, opaque["event"], {"dev": dev})
|
||
|
|
||
|
|
||
|
def _domain_event_tunable_cb(conn, domain, params, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain tunable events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'params': params
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(opaque, conn, domain, opaque["event"], {"params": params})
|
||
|
|
||
|
|
||
|
# pylint: disable=invalid-name
|
||
|
def _domain_event_agent_lifecycle_cb(conn, domain, state, reason, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain agent lifecycle events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'state': _get_libvirt_enum_string('VIR_CONNECT_DOMAIN_EVENT_AGENT_LIFECYCLE_STATE_', state),
|
||
|
- 'reason': _get_libvirt_enum_string('VIR_CONNECT_DOMAIN_EVENT_AGENT_LIFECYCLE_REASON_', reason)
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ domain,
|
||
|
+ opaque["event"],
|
||
|
+ {
|
||
|
+ "state": _get_libvirt_enum_string(
|
||
|
+ "VIR_CONNECT_DOMAIN_EVENT_AGENT_LIFECYCLE_STATE_", state
|
||
|
+ ),
|
||
|
+ "reason": _get_libvirt_enum_string(
|
||
|
+ "VIR_CONNECT_DOMAIN_EVENT_AGENT_LIFECYCLE_REASON_", reason
|
||
|
+ ),
|
||
|
+ },
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_device_added_cb(conn, domain, dev, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain device addition events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'dev': dev
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(opaque, conn, domain, opaque["event"], {"dev": dev})
|
||
|
|
||
|
|
||
|
# pylint: disable=invalid-name
|
||
|
def _domain_event_migration_iteration_cb(conn, domain, iteration, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain migration iteration events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'iteration': iteration
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque, conn, domain, opaque["event"], {"iteration": iteration}
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _domain_event_job_completed_cb(conn, domain, params, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain job completion events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'params': params
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(opaque, conn, domain, opaque["event"], {"params": params})
|
||
|
|
||
|
|
||
|
def _domain_event_device_removal_failed_cb(conn, domain, dev, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain device removal failure events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'dev': dev
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(opaque, conn, domain, opaque["event"], {"dev": dev})
|
||
|
|
||
|
|
||
|
def _domain_event_metadata_change_cb(conn, domain, mtype, nsuri, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Domain metadata change events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'type': _get_libvirt_enum_string('VIR_DOMAIN_METADATA_', mtype),
|
||
|
- 'nsuri': nsuri
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ domain,
|
||
|
+ opaque["event"],
|
||
|
+ {
|
||
|
+ "type": _get_libvirt_enum_string("VIR_DOMAIN_METADATA_", mtype),
|
||
|
+ "nsuri": nsuri,
|
||
|
+ },
|
||
|
+ )
|
||
|
|
||
|
|
||
|
-def _domain_event_block_threshold_cb(conn, domain, dev, path, threshold, excess, opaque):
|
||
|
- '''
|
||
|
+def _domain_event_block_threshold_cb(
|
||
|
+ conn, domain, dev, path, threshold, excess, opaque
|
||
|
+):
|
||
|
+ """
|
||
|
Domain block threshold events handler
|
||
|
- '''
|
||
|
- _salt_send_domain_event(opaque, conn, domain, opaque['event'], {
|
||
|
- 'dev': dev,
|
||
|
- 'path': path,
|
||
|
- 'threshold': threshold,
|
||
|
- 'excess': excess
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_domain_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ domain,
|
||
|
+ opaque["event"],
|
||
|
+ {"dev": dev, "path": path, "threshold": threshold, "excess": excess},
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _network_event_lifecycle_cb(conn, net, event, detail, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Network lifecycle events handler
|
||
|
- '''
|
||
|
-
|
||
|
- _salt_send_event(opaque, conn, {
|
||
|
- 'network': {
|
||
|
- 'name': net.name(),
|
||
|
- 'uuid': net.UUIDString()
|
||
|
+ """
|
||
|
+
|
||
|
+ _salt_send_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ {
|
||
|
+ "network": {"name": net.name(), "uuid": net.UUIDString()},
|
||
|
+ "event": _get_libvirt_enum_string("VIR_NETWORK_EVENT_", event),
|
||
|
+ "detail": "unknown", # currently unused
|
||
|
},
|
||
|
- 'event': _get_libvirt_enum_string('VIR_NETWORK_EVENT_', event),
|
||
|
- 'detail': 'unknown' # currently unused
|
||
|
- })
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _pool_event_lifecycle_cb(conn, pool, event, detail, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Storage pool lifecycle events handler
|
||
|
- '''
|
||
|
- _salt_send_event(opaque, conn, {
|
||
|
- 'pool': {
|
||
|
- 'name': pool.name(),
|
||
|
- 'uuid': pool.UUIDString()
|
||
|
+ """
|
||
|
+ _salt_send_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ {
|
||
|
+ "pool": {"name": pool.name(), "uuid": pool.UUIDString()},
|
||
|
+ "event": _get_libvirt_enum_string("VIR_STORAGE_POOL_EVENT_", event),
|
||
|
+ "detail": "unknown", # currently unused
|
||
|
},
|
||
|
- 'event': _get_libvirt_enum_string('VIR_STORAGE_POOL_EVENT_', event),
|
||
|
- 'detail': 'unknown' # currently unused
|
||
|
- })
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _pool_event_refresh_cb(conn, pool, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Storage pool refresh events handler
|
||
|
- '''
|
||
|
- _salt_send_event(opaque, conn, {
|
||
|
- 'pool': {
|
||
|
- 'name': pool.name(),
|
||
|
- 'uuid': pool.UUIDString()
|
||
|
+ """
|
||
|
+ _salt_send_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ {
|
||
|
+ "pool": {"name": pool.name(), "uuid": pool.UUIDString()},
|
||
|
+ "event": opaque["event"],
|
||
|
},
|
||
|
- 'event': opaque['event']
|
||
|
- })
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _nodedev_event_lifecycle_cb(conn, dev, event, detail, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Node device lifecycle events handler
|
||
|
- '''
|
||
|
- _salt_send_event(opaque, conn, {
|
||
|
- 'nodedev': {
|
||
|
- 'name': dev.name()
|
||
|
+ """
|
||
|
+ _salt_send_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ {
|
||
|
+ "nodedev": {"name": dev.name()},
|
||
|
+ "event": _get_libvirt_enum_string("VIR_NODE_DEVICE_EVENT_", event),
|
||
|
+ "detail": "unknown", # currently unused
|
||
|
},
|
||
|
- 'event': _get_libvirt_enum_string('VIR_NODE_DEVICE_EVENT_', event),
|
||
|
- 'detail': 'unknown' # currently unused
|
||
|
- })
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _nodedev_event_update_cb(conn, dev, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Node device update events handler
|
||
|
- '''
|
||
|
- _salt_send_event(opaque, conn, {
|
||
|
- 'nodedev': {
|
||
|
- 'name': dev.name()
|
||
|
- },
|
||
|
- 'event': opaque['event']
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_event(
|
||
|
+ opaque, conn, {"nodedev": {"name": dev.name()}, "event": opaque["event"]}
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _secret_event_lifecycle_cb(conn, secret, event, detail, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Secret lifecycle events handler
|
||
|
- '''
|
||
|
- _salt_send_event(opaque, conn, {
|
||
|
- 'secret': {
|
||
|
- 'uuid': secret.UUIDString()
|
||
|
+ """
|
||
|
+ _salt_send_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ {
|
||
|
+ "secret": {"uuid": secret.UUIDString()},
|
||
|
+ "event": _get_libvirt_enum_string("VIR_SECRET_EVENT_", event),
|
||
|
+ "detail": "unknown", # currently unused
|
||
|
},
|
||
|
- 'event': _get_libvirt_enum_string('VIR_SECRET_EVENT_', event),
|
||
|
- 'detail': 'unknown' # currently unused
|
||
|
- })
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _secret_event_value_changed_cb(conn, secret, opaque):
|
||
|
- '''
|
||
|
+ """
|
||
|
Secret value change events handler
|
||
|
- '''
|
||
|
- _salt_send_event(opaque, conn, {
|
||
|
- 'secret': {
|
||
|
- 'uuid': secret.UUIDString()
|
||
|
- },
|
||
|
- 'event': opaque['event']
|
||
|
- })
|
||
|
+ """
|
||
|
+ _salt_send_event(
|
||
|
+ opaque,
|
||
|
+ conn,
|
||
|
+ {"secret": {"uuid": secret.UUIDString()}, "event": opaque["event"]},
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _cleanup(cnx):
|
||
|
- '''
|
||
|
+ """
|
||
|
Close the libvirt connection
|
||
|
|
||
|
:param cnx: libvirt connection
|
||
|
- '''
|
||
|
- log.debug('Closing libvirt connection: %s', cnx.getURI())
|
||
|
+ """
|
||
|
+ log.debug("Closing libvirt connection: %s", cnx.getURI())
|
||
|
cnx.close()
|
||
|
|
||
|
|
||
|
def _callbacks_cleanup(cnx, callback_ids):
|
||
|
- '''
|
||
|
+ """
|
||
|
Unregister all the registered callbacks
|
||
|
|
||
|
:param cnx: libvirt connection
|
||
|
:param callback_ids: dictionary mapping a libvirt object type to an ID list
|
||
|
of callbacks to deregister
|
||
|
- '''
|
||
|
+ """
|
||
|
for obj, ids in callback_ids.items():
|
||
|
register_name = REGISTER_FUNCTIONS[obj]
|
||
|
- deregister_name = register_name.replace('Reg', 'Dereg')
|
||
|
+ deregister_name = register_name.replace("Reg", "Dereg")
|
||
|
deregister = getattr(cnx, deregister_name)
|
||
|
for callback_id in ids:
|
||
|
deregister(callback_id)
|
||
|
|
||
|
|
||
|
def _register_callback(cnx, tag_prefix, obj, event, real_id):
|
||
|
- '''
|
||
|
+ """
|
||
|
Helper function registering a callback
|
||
|
|
||
|
:param cnx: libvirt connection
|
||
|
@@ -620,10 +685,10 @@ def _register_callback(cnx, tag_prefix, obj, event, real_id):
|
||
|
:param real_id: the libvirt name of an alternative event id to use or None
|
||
|
|
||
|
:rtype integer value needed to deregister the callback
|
||
|
- '''
|
||
|
+ """
|
||
|
libvirt_name = real_id
|
||
|
if real_id is None:
|
||
|
- libvirt_name = 'VIR_{0}_EVENT_ID_{1}'.format(obj, event).upper()
|
||
|
+ libvirt_name = "VIR_{0}_EVENT_ID_{1}".format(obj, event).upper()
|
||
|
|
||
|
if not hasattr(libvirt, libvirt_name):
|
||
|
log.warning('Skipping "%s/%s" events: libvirt too old', obj, event)
|
||
|
@@ -633,34 +698,34 @@ def _register_callback(cnx, tag_prefix, obj, event, real_id):
|
||
|
callback_name = "_{0}_event_{1}_cb".format(obj, event)
|
||
|
callback = globals().get(callback_name, None)
|
||
|
if callback is None:
|
||
|
- log.error('Missing function %s in engine', callback_name)
|
||
|
+ log.error("Missing function %s in engine", callback_name)
|
||
|
return None
|
||
|
|
||
|
register = getattr(cnx, REGISTER_FUNCTIONS[obj])
|
||
|
- return register(None, libvirt_id, callback,
|
||
|
- {'prefix': tag_prefix,
|
||
|
- 'object': obj,
|
||
|
- 'event': event})
|
||
|
+ return register(
|
||
|
+ None,
|
||
|
+ libvirt_id,
|
||
|
+ callback,
|
||
|
+ {"prefix": tag_prefix, "object": obj, "event": event},
|
||
|
+ )
|
||
|
|
||
|
|
||
|
def _append_callback_id(ids, obj, callback_id):
|
||
|
- '''
|
||
|
+ """
|
||
|
Helper function adding a callback ID to the IDs dict.
|
||
|
The callback ids dict maps an object to event callback ids.
|
||
|
|
||
|
:param ids: dict of callback IDs to update
|
||
|
:param obj: one of the keys of REGISTER_FUNCTIONS
|
||
|
:param callback_id: the result of _register_callback
|
||
|
- '''
|
||
|
+ """
|
||
|
if obj not in ids:
|
||
|
ids[obj] = []
|
||
|
ids[obj].append(callback_id)
|
||
|
|
||
|
|
||
|
-def start(uri=None,
|
||
|
- tag_prefix='salt/engines/libvirt_events',
|
||
|
- filters=None):
|
||
|
- '''
|
||
|
+def start(uri=None, tag_prefix="salt/engines/libvirt_events", filters=None):
|
||
|
+ """
|
||
|
Listen to libvirt events and forward them to salt.
|
||
|
|
||
|
:param uri: libvirt URI to listen on.
|
||
|
@@ -668,14 +733,14 @@ def start(uri=None,
|
||
|
:param tag_prefix: the begining of the salt event tag to use.
|
||
|
Defaults to 'salt/engines/libvirt_events'
|
||
|
:param filters: the list of event of listen on. Defaults to 'all'
|
||
|
- '''
|
||
|
+ """
|
||
|
if filters is None:
|
||
|
- filters = ['all']
|
||
|
+ filters = ["all"]
|
||
|
try:
|
||
|
libvirt.virEventRegisterDefaultImpl()
|
||
|
|
||
|
cnx = libvirt.openReadOnly(uri)
|
||
|
- log.debug('Opened libvirt uri: %s', cnx.getURI())
|
||
|
+ log.debug("Opened libvirt uri: %s", cnx.getURI())
|
||
|
|
||
|
callback_ids = {}
|
||
|
all_filters = "all" in filters
|
||
|
@@ -683,17 +748,19 @@ def start(uri=None,
|
||
|
for obj, event_defs in CALLBACK_DEFS.items():
|
||
|
for event, real_id in event_defs:
|
||
|
event_filter = "/".join((obj, event))
|
||
|
- if event_filter not in filters and obj not in filters and not all_filters:
|
||
|
+ if (
|
||
|
+ event_filter not in filters
|
||
|
+ and obj not in filters
|
||
|
+ and not all_filters
|
||
|
+ ):
|
||
|
continue
|
||
|
- registered_id = _register_callback(cnx, tag_prefix,
|
||
|
- obj, event, real_id)
|
||
|
+ registered_id = _register_callback(cnx, tag_prefix, obj, event, real_id)
|
||
|
if registered_id:
|
||
|
_append_callback_id(callback_ids, obj, registered_id)
|
||
|
|
||
|
exit_loop = False
|
||
|
while not exit_loop:
|
||
|
exit_loop = libvirt.virEventRunDefaultImpl() < 0
|
||
|
- log.debug('=== in the loop exit_loop %s ===', exit_loop)
|
||
|
|
||
|
except Exception as err: # pylint: disable=broad-except
|
||
|
log.exception(err)
|
||
|
diff --git a/tests/unit/engines/test_libvirt_events.py b/tests/unit/engines/test_libvirt_events.py
|
||
|
index d9143a320b..5f1488e422 100644
|
||
|
--- a/tests/unit/engines/test_libvirt_events.py
|
||
|
+++ b/tests/unit/engines/test_libvirt_events.py
|
||
|
@@ -1,16 +1,14 @@
|
||
|
# -*- coding: utf-8 -*-
|
||
|
-'''
|
||
|
+"""
|
||
|
unit tests for the libvirt_events engine
|
||
|
-'''
|
||
|
+"""
|
||
|
# Import Python libs
|
||
|
from __future__ import absolute_import, print_function, unicode_literals
|
||
|
|
||
|
# Import Salt Testing Libs
|
||
|
from tests.support.mixins import LoaderModuleMockMixin
|
||
|
from tests.support.unit import TestCase
|
||
|
-from tests.support.mock import (
|
||
|
- MagicMock,
|
||
|
- patch)
|
||
|
+from tests.support.mock import MagicMock, patch
|
||
|
|
||
|
# Import Salt Libs
|
||
|
import salt.engines.libvirt_events as libvirt_events
|
||
|
@@ -20,68 +18,78 @@ import salt.engines.libvirt_events as libvirt_events
|
||
|
|
||
|
|
||
|
class EngineLibvirtEventTestCase(TestCase, LoaderModuleMockMixin):
|
||
|
- '''
|
||
|
+ """
|
||
|
Test cases for salt.engine.libvirt_events
|
||
|
- '''
|
||
|
+ """
|
||
|
|
||
|
def setup_loader_modules(self):
|
||
|
- patcher = patch('salt.engines.libvirt_events.libvirt')
|
||
|
+ patcher = patch("salt.engines.libvirt_events.libvirt")
|
||
|
self.mock_libvirt = patcher.start()
|
||
|
self.mock_libvirt.getVersion.return_value = 2000000
|
||
|
- self.mock_libvirt.virEventRunDefaultImpl.return_value = -1 # Don't loop for ever
|
||
|
+ self.mock_libvirt.virEventRunDefaultImpl.return_value = (
|
||
|
+ -1
|
||
|
+ ) # Don't loop for ever
|
||
|
self.mock_libvirt.VIR_DOMAIN_EVENT_ID_LIFECYCLE = 0
|
||
|
self.mock_libvirt.VIR_DOMAIN_EVENT_ID_REBOOT = 1
|
||
|
+ self.mock_libvirt.VIR_STORAGE_POOL_EVENT_ID_LIFECYCLE = 0
|
||
|
+ self.mock_libvirt.VIR_STORAGE_POOL_EVENT_ID_REFRESH = 1
|
||
|
+ self.mock_libvirt.VIR_NODE_DEVICE_EVENT_ID_LIFECYCLE = 0
|
||
|
+ self.mock_libvirt.VIR_NODE_DEVICE_EVENT_ID_UPDATE = 1
|
||
|
self.addCleanup(patcher.stop)
|
||
|
- self.addCleanup(delattr, self, 'mock_libvirt')
|
||
|
+ self.addCleanup(delattr, self, "mock_libvirt")
|
||
|
return {libvirt_events: {}}
|
||
|
|
||
|
- @patch('salt.engines.libvirt_events.libvirt',
|
||
|
- VIR_PREFIX_NONE=0,
|
||
|
- VIR_PREFIX_ONE=1,
|
||
|
- VIR_PREFIX_TWO=2,
|
||
|
- VIR_PREFIX_SUB_FOO=0,
|
||
|
- VIR_PREFIX_SUB_BAR=1,
|
||
|
- VIR_PREFIX_SUB_FOOBAR=2)
|
||
|
+ @patch(
|
||
|
+ "salt.engines.libvirt_events.libvirt",
|
||
|
+ VIR_PREFIX_NONE=0,
|
||
|
+ VIR_PREFIX_ONE=1,
|
||
|
+ VIR_PREFIX_TWO=2,
|
||
|
+ VIR_PREFIX_SUB_FOO=0,
|
||
|
+ VIR_PREFIX_SUB_BAR=1,
|
||
|
+ VIR_PREFIX_SUB_FOOBAR=2,
|
||
|
+ )
|
||
|
def test_get_libvirt_enum_string_subprefix(self, libvirt_mock):
|
||
|
- '''
|
||
|
+ """
|
||
|
Make sure the libvirt enum value to string works reliably with
|
||
|
elements with a sub prefix, eg VIR_PREFIX_SUB_* in this case.
|
||
|
- '''
|
||
|
+ """
|
||
|
# Test case with a sub prefix
|
||
|
|
||
|
- assert libvirt_events._get_libvirt_enum_string('VIR_PREFIX_', 2) == 'two'
|
||
|
+ assert libvirt_events._get_libvirt_enum_string("VIR_PREFIX_", 2) == "two"
|
||
|
|
||
|
- @patch('salt.engines.libvirt_events.libvirt',
|
||
|
- VIR_PREFIX_FOO=0,
|
||
|
- VIR_PREFIX_BAR_FOO=1)
|
||
|
+ @patch(
|
||
|
+ "salt.engines.libvirt_events.libvirt", VIR_PREFIX_FOO=0, VIR_PREFIX_BAR_FOO=1
|
||
|
+ )
|
||
|
def test_get_libvirt_enum_string_underscores(self, libvirt_mock):
|
||
|
- '''
|
||
|
+ """
|
||
|
Make sure the libvirt enum value to string works reliably and items
|
||
|
with an underscore aren't confused with sub prefixes.
|
||
|
- '''
|
||
|
- assert libvirt_events._get_libvirt_enum_string('VIR_PREFIX_', 1) == 'bar foo'
|
||
|
-
|
||
|
- @patch('salt.engines.libvirt_events.libvirt',
|
||
|
- VIR_DOMAIN_EVENT_CRASHED_PANICKED=0,
|
||
|
- VIR_DOMAIN_EVENT_DEFINED=0,
|
||
|
- VIR_DOMAIN_EVENT_UNDEFINED=1,
|
||
|
- VIR_DOMAIN_EVENT_CRASHED=2,
|
||
|
- VIR_DOMAIN_EVENT_DEFINED_ADDED=0,
|
||
|
- VIR_DOMAIN_EVENT_DEFINED_UPDATED=1)
|
||
|
+ """
|
||
|
+ assert libvirt_events._get_libvirt_enum_string("VIR_PREFIX_", 1) == "bar foo"
|
||
|
+
|
||
|
+ @patch(
|
||
|
+ "salt.engines.libvirt_events.libvirt",
|
||
|
+ VIR_DOMAIN_EVENT_CRASHED_PANICKED=0,
|
||
|
+ VIR_DOMAIN_EVENT_DEFINED=0,
|
||
|
+ VIR_DOMAIN_EVENT_UNDEFINED=1,
|
||
|
+ VIR_DOMAIN_EVENT_CRASHED=2,
|
||
|
+ VIR_DOMAIN_EVENT_DEFINED_ADDED=0,
|
||
|
+ VIR_DOMAIN_EVENT_DEFINED_UPDATED=1,
|
||
|
+ )
|
||
|
def test_get_domain_event_detail(self, mock_libvirt):
|
||
|
- '''
|
||
|
+ """
|
||
|
Test get_domain_event_detail function
|
||
|
- '''
|
||
|
- assert libvirt_events._get_domain_event_detail(1, 2) == ('undefined', 'unknown')
|
||
|
- assert libvirt_events._get_domain_event_detail(0, 1) == ('defined', 'updated')
|
||
|
- assert libvirt_events._get_domain_event_detail(4, 2) == ('unknown', 'unknown')
|
||
|
+ """
|
||
|
+ assert libvirt_events._get_domain_event_detail(1, 2) == ("undefined", "unknown")
|
||
|
+ assert libvirt_events._get_domain_event_detail(0, 1) == ("defined", "updated")
|
||
|
+ assert libvirt_events._get_domain_event_detail(4, 2) == ("unknown", "unknown")
|
||
|
|
||
|
- @patch('salt.engines.libvirt_events.libvirt', VIR_NETWORK_EVENT_ID_LIFECYCLE=1000)
|
||
|
+ @patch("salt.engines.libvirt_events.libvirt", VIR_NETWORK_EVENT_ID_LIFECYCLE=1000)
|
||
|
def test_event_register(self, mock_libvirt):
|
||
|
- '''
|
||
|
+ """
|
||
|
Test that the libvirt_events engine actually registers events catch them and cleans
|
||
|
before leaving the place.
|
||
|
- '''
|
||
|
+ """
|
||
|
mock_cnx = MagicMock()
|
||
|
mock_libvirt.openReadOnly.return_value = mock_cnx
|
||
|
|
||
|
@@ -90,71 +98,112 @@ class EngineLibvirtEventTestCase(TestCase, LoaderModuleMockMixin):
|
||
|
|
||
|
mock_cnx.networkEventRegisterAny.return_value = 10000
|
||
|
|
||
|
- libvirt_events.start('test:///', 'test/prefix')
|
||
|
+ libvirt_events.start("test:///", "test/prefix")
|
||
|
|
||
|
# Check that the connection has been opened
|
||
|
- mock_libvirt.openReadOnly.assert_called_once_with('test:///')
|
||
|
+ mock_libvirt.openReadOnly.assert_called_once_with("test:///")
|
||
|
|
||
|
# Check that the connection has been closed
|
||
|
mock_cnx.close.assert_called_once()
|
||
|
|
||
|
# Check events registration and deregistration
|
||
|
mock_cnx.domainEventRegisterAny.assert_any_call(
|
||
|
- None, mock_libvirt.VIR_DOMAIN_EVENT_ID_LIFECYCLE,
|
||
|
- libvirt_events._domain_event_lifecycle_cb,
|
||
|
- {'prefix': 'test/prefix', 'object': 'domain', 'event': 'lifecycle'})
|
||
|
+ None,
|
||
|
+ mock_libvirt.VIR_DOMAIN_EVENT_ID_LIFECYCLE,
|
||
|
+ libvirt_events._domain_event_lifecycle_cb,
|
||
|
+ {"prefix": "test/prefix", "object": "domain", "event": "lifecycle"},
|
||
|
+ )
|
||
|
mock_cnx.networkEventRegisterAny.assert_any_call(
|
||
|
- None, mock_libvirt.VIR_NETWORK_EVENT_ID_LIFECYCLE,
|
||
|
- libvirt_events._network_event_lifecycle_cb,
|
||
|
- {'prefix': 'test/prefix', 'object': 'network', 'event': 'lifecycle'})
|
||
|
+ None,
|
||
|
+ mock_libvirt.VIR_NETWORK_EVENT_ID_LIFECYCLE,
|
||
|
+ libvirt_events._network_event_lifecycle_cb,
|
||
|
+ {"prefix": "test/prefix", "object": "network", "event": "lifecycle"},
|
||
|
+ )
|
||
|
+ mock_cnx.storagePoolEventRegisterAny.assert_any_call(
|
||
|
+ None,
|
||
|
+ mock_libvirt.VIR_STORAGE_POOL_EVENT_ID_LIFECYCLE,
|
||
|
+ libvirt_events._pool_event_lifecycle_cb,
|
||
|
+ {"prefix": "test/prefix", "object": "pool", "event": "lifecycle"},
|
||
|
+ )
|
||
|
+ mock_cnx.storagePoolEventRegisterAny.assert_any_call(
|
||
|
+ None,
|
||
|
+ mock_libvirt.VIR_STORAGE_POOL_EVENT_ID_REFRESH,
|
||
|
+ libvirt_events._pool_event_refresh_cb,
|
||
|
+ {"prefix": "test/prefix", "object": "pool", "event": "refresh"},
|
||
|
+ )
|
||
|
+ mock_cnx.nodeDeviceEventRegisterAny.assert_any_call(
|
||
|
+ None,
|
||
|
+ mock_libvirt.VIR_NODE_DEVICE_EVENT_ID_LIFECYCLE,
|
||
|
+ libvirt_events._nodedev_event_lifecycle_cb,
|
||
|
+ {"prefix": "test/prefix", "object": "nodedev", "event": "lifecycle"},
|
||
|
+ )
|
||
|
+ mock_cnx.nodeDeviceEventRegisterAny.assert_any_call(
|
||
|
+ None,
|
||
|
+ mock_libvirt.VIR_NODE_DEVICE_EVENT_ID_UPDATE,
|
||
|
+ libvirt_events._nodedev_event_update_cb,
|
||
|
+ {"prefix": "test/prefix", "object": "nodedev", "event": "update"},
|
||
|
+ )
|
||
|
|
||
|
# Check that the deregister events are called with the result of register
|
||
|
mock_cnx.networkEventDeregisterAny.assert_called_with(
|
||
|
- mock_cnx.networkEventRegisterAny.return_value)
|
||
|
+ mock_cnx.networkEventRegisterAny.return_value
|
||
|
+ )
|
||
|
|
||
|
# Check that the default 'all' filter actually worked
|
||
|
- counts = {obj: len(callback_def) for obj, callback_def in libvirt_events.CALLBACK_DEFS.items()}
|
||
|
+ counts = {
|
||
|
+ obj: len(callback_def)
|
||
|
+ for obj, callback_def in libvirt_events.CALLBACK_DEFS.items()
|
||
|
+ }
|
||
|
for obj, count in counts.items():
|
||
|
register = libvirt_events.REGISTER_FUNCTIONS[obj]
|
||
|
assert getattr(mock_cnx, register).call_count == count
|
||
|
|
||
|
def test_event_skipped(self):
|
||
|
- '''
|
||
|
+ """
|
||
|
Test that events are skipped if their ID isn't defined in the libvirt
|
||
|
module (older libvirt)
|
||
|
- '''
|
||
|
- self.mock_libvirt.mock_add_spec([
|
||
|
- 'openReadOnly',
|
||
|
- 'virEventRegisterDefaultImpl',
|
||
|
- 'virEventRunDefaultImpl',
|
||
|
- 'VIR_DOMAIN_EVENT_ID_LIFECYCLE'], spec_set=True)
|
||
|
-
|
||
|
- libvirt_events.start('test:///', 'test/prefix')
|
||
|
+ """
|
||
|
+ self.mock_libvirt.mock_add_spec(
|
||
|
+ [
|
||
|
+ "openReadOnly",
|
||
|
+ "virEventRegisterDefaultImpl",
|
||
|
+ "virEventRunDefaultImpl",
|
||
|
+ "VIR_DOMAIN_EVENT_ID_LIFECYCLE",
|
||
|
+ ],
|
||
|
+ spec_set=True,
|
||
|
+ )
|
||
|
+
|
||
|
+ libvirt_events.start("test:///", "test/prefix")
|
||
|
|
||
|
# Check events registration and deregistration
|
||
|
mock_cnx = self.mock_libvirt.openReadOnly.return_value
|
||
|
|
||
|
mock_cnx.domainEventRegisterAny.assert_any_call(
|
||
|
- None, self.mock_libvirt.VIR_DOMAIN_EVENT_ID_LIFECYCLE,
|
||
|
- libvirt_events._domain_event_lifecycle_cb,
|
||
|
- {'prefix': 'test/prefix', 'object': 'domain', 'event': 'lifecycle'})
|
||
|
+ None,
|
||
|
+ self.mock_libvirt.VIR_DOMAIN_EVENT_ID_LIFECYCLE,
|
||
|
+ libvirt_events._domain_event_lifecycle_cb,
|
||
|
+ {"prefix": "test/prefix", "object": "domain", "event": "lifecycle"},
|
||
|
+ )
|
||
|
|
||
|
# Network events should have been skipped
|
||
|
mock_cnx.networkEventRegisterAny.assert_not_called()
|
||
|
|
||
|
def test_event_filtered(self):
|
||
|
- '''
|
||
|
+ """
|
||
|
Test that events are skipped if their ID isn't defined in the libvirt
|
||
|
module (older libvirt)
|
||
|
- '''
|
||
|
- libvirt_events.start('test', 'test/prefix', 'domain/lifecycle')
|
||
|
+ """
|
||
|
+ libvirt_events.start("test", "test/prefix", "domain/lifecycle")
|
||
|
|
||
|
# Check events registration and deregistration
|
||
|
mock_cnx = self.mock_libvirt.openReadOnly.return_value
|
||
|
|
||
|
mock_cnx.domainEventRegisterAny.assert_any_call(
|
||
|
- None, 0, libvirt_events._domain_event_lifecycle_cb,
|
||
|
- {'prefix': 'test/prefix', 'object': 'domain', 'event': 'lifecycle'})
|
||
|
+ None,
|
||
|
+ 0,
|
||
|
+ libvirt_events._domain_event_lifecycle_cb,
|
||
|
+ {"prefix": "test/prefix", "object": "domain", "event": "lifecycle"},
|
||
|
+ )
|
||
|
|
||
|
# Network events should have been filtered out
|
||
|
mock_cnx.networkEventRegisterAny.assert_not_called()
|
||
|
--
|
||
|
2.27.0
|
||
|
|
||
|
|