mirror of
https://github.com/virt-manager/virt-manager.git
synced 2025-01-03 01:18:00 +03:00
4a2df06483
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>
292 lines
8.4 KiB
Python
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())
|