virt-manager/virtManager/object/storagepool.py
Lin Ma 4a2df06483 diskbackend: Drop support for sheepdog
The sheepdog project is no longer actively developed, Libvirt removed
the support for sheepdog storage backend since v8.8.0, Let's drop it.

Signed-off-by: Lin Ma <lma@suse.com>
2022-11-07 10:10:00 -05:00

292 lines
8.4 KiB
Python

# Copyright (C) 2008, 2013 Red Hat, Inc.
# Copyright (C) 2008 Cole Robinson <crobinso@redhat.com>
#
# This work is licensed under the GNU GPLv2 or later.
# See the COPYING file in the top-level directory.
import time
from virtinst import log
from virtinst import pollhelpers
from virtinst import StoragePool, StorageVolume
from .libvirtobject import vmmLibvirtObject
def _pretty_bytes(val):
val = int(val)
if val > (1024 * 1024 * 1024):
return "%2.2f GiB" % (val / (1024.0 * 1024.0 * 1024.0))
else:
return "%2.2f MiB" % (val / (1024.0 * 1024.0))
POOL_TYPE_DESCS = {
StoragePool.TYPE_DIR: _("Filesystem Directory"),
StoragePool.TYPE_FS: _("Pre-Formatted Block Device"),
StoragePool.TYPE_NETFS: _("Network Exported Directory"),
StoragePool.TYPE_LOGICAL: _("LVM Volume Group"),
StoragePool.TYPE_DISK: _("Physical Disk Device"),
StoragePool.TYPE_ISCSI: _("iSCSI Target"),
StoragePool.TYPE_SCSI: _("SCSI Host Adapter"),
StoragePool.TYPE_MPATH: _("Multipath Device Enumerator"),
StoragePool.TYPE_GLUSTER: _("Gluster Filesystem"),
StoragePool.TYPE_RBD: _("RADOS Block Device/Ceph"),
StoragePool.TYPE_ZFS: _("ZFS Pool"),
}
class vmmStorageVolume(vmmLibvirtObject):
def __init__(self, conn, backend, key):
vmmLibvirtObject.__init__(self, conn, backend, key, StorageVolume)
##########################
# Required class methods #
##########################
def _conn_tick_poll_param(self):
return None # pragma: no cover
def class_name(self):
return "volume" # pragma: no cover
def _XMLDesc(self, flags):
try:
return self._backend.XMLDesc(flags)
except Exception as e: # pragma: no cover
log.debug("XMLDesc for vol=%s failed: %s",
self._backend.key(), e)
raise
def _get_backend_status(self):
return self._STATUS_ACTIVE
###########
# Actions #
###########
def get_parent_pool(self):
name = self._backend.storagePoolLookupByVolume().name()
for pool in self.conn.list_pools():
if pool.get_name() == name:
return pool
def delete(self, force=True):
ignore = force
self._backend.delete(0)
self._backend = None
#################
# XML accessors #
#################
def get_key(self):
return self.get_xmlobj().key or ""
def get_target_path(self):
return self.get_xmlobj().target_path or ""
def get_format(self):
return self.get_xmlobj().format
def get_capacity(self):
return self.get_xmlobj().capacity
def get_pretty_capacity(self):
return _pretty_bytes(self.get_capacity())
def get_pretty_name(self, pooltype):
name = self.get_name()
if pooltype != "iscsi":
return name
key = self.get_key()
ret = name
if key:
ret += " (%s)" % key
return ret
class vmmStoragePool(vmmLibvirtObject):
__gsignals__ = {
"refreshed": (vmmLibvirtObject.RUN_FIRST, None, [])
}
@staticmethod
def supports_volume_creation(pool_type, clone=False):
"""
Returns if pool supports volume creation. If @clone is set to True
returns if pool supports volume cloning (virVolCreateXMLFrom).
"""
supported = [
StoragePool.TYPE_DIR,
StoragePool.TYPE_FS,
StoragePool.TYPE_NETFS,
StoragePool.TYPE_DISK,
StoragePool.TYPE_LOGICAL,
StoragePool.TYPE_RBD,
]
if not clone:
supported.extend([
StoragePool.TYPE_ZFS,
])
return pool_type in supported
@staticmethod
def pretty_type(pool_type):
return POOL_TYPE_DESCS.get(pool_type, "%s pool" % pool_type)
@staticmethod
def list_types():
return sorted(list(POOL_TYPE_DESCS.keys()))
def __init__(self, conn, backend, key):
vmmLibvirtObject.__init__(self, conn, backend, key, StoragePool)
self._last_refresh_time = 0
self._volumes = None
##########################
# Required class methods #
##########################
def _conn_tick_poll_param(self):
return "pollpool"
def class_name(self):
return "pool"
def _XMLDesc(self, flags):
return self._backend.XMLDesc(flags)
def _define(self, xml):
return self.conn.define_pool(xml)
def _using_events(self):
return self.conn.using_storage_pool_events
def _get_backend_status(self):
return (bool(self._backend.isActive()) and
self._STATUS_ACTIVE or
self._STATUS_INACTIVE)
def _init_libvirt_state(self):
super()._init_libvirt_state()
if not self.conn.is_active():
# We only want to refresh a pool on initial conn startup,
# since the pools may be out of date. But if a storage pool
# shows up while the conn is connected, this means it was
# just 'defined' recently and doesn't need to be refreshed.
self.refresh(_from_object_init=True)
for vol in self.get_volumes():
vol.init_libvirt_state()
def _invalidate_xml(self):
vmmLibvirtObject._invalidate_xml(self)
self._volumes = None
def _cleanup(self):
vmmLibvirtObject._cleanup(self)
for vol in (self._volumes or []):
vol.cleanup()
self._volumes = None
###########
# Actions #
###########
@vmmLibvirtObject.lifecycle_action
def start(self):
self._backend.create(0)
@vmmLibvirtObject.lifecycle_action
def stop(self):
self._backend.destroy()
@vmmLibvirtObject.lifecycle_action
def delete(self, force=True):
ignore = force
self._backend.undefine()
self._backend = None
def refresh(self, _from_object_init=False):
"""
:param _from_object_init: Only used for the refresh() call from
_init_libvirt_state. Tells us to not refresh the XML, since
we just updated it.
"""
if not self.is_active():
return # pragma: no cover
self._backend.refresh(0)
if self._using_events() and not _from_object_init:
# If we are using events, we let the event loop trigger
# the cache update for us. Except if from init_libvirt_state,
# we want the update to be done immediately
return
self.refresh_pool_cache_from_event_loop(
_from_object_init=_from_object_init)
def refresh_pool_cache_from_event_loop(self, _from_object_init=False):
if not _from_object_init:
self.recache_from_event_loop()
self._update_volumes(force=True)
self.idle_emit("refreshed")
self._last_refresh_time = time.time()
def secs_since_last_refresh(self):
return time.time() - self._last_refresh_time
###################
# Volume handling #
###################
def get_volume_by_name(self, name):
for vol in self.get_volumes():
if vol.get_name() == name:
return vol
def get_volumes(self):
self._update_volumes(force=False)
return self._volumes[:]
def _update_volumes(self, force):
if not self.is_active():
self._volumes = []
return
if not force and self._volumes is not None:
return
keymap = dict((o.get_name(), o) for o in self._volumes or [])
def cb(obj, key):
return vmmStorageVolume(self.conn, obj, key)
(dummy1, dummy2, allvols) = pollhelpers.fetch_volumes(
self.conn.get_backend(), self.get_backend(), keymap, cb)
self._volumes = allvols
#########################
# XML/config operations #
#########################
def set_autostart(self, val):
self._backend.setAutostart(val)
def get_autostart(self):
return self._backend.autostart()
def get_type(self):
return self.get_xmlobj().type
def get_target_path(self):
return self.get_xmlobj().target_path or ""
def get_allocation(self):
return self.get_xmlobj().allocation
def get_available(self):
return self.get_xmlobj().available
def get_capacity(self):
return self.get_xmlobj().capacity
def get_pretty_allocation(self):
return _pretty_bytes(self.get_allocation())
def get_pretty_available(self):
return _pretty_bytes(self.get_available())