qubes.storage rework api
This commit is contained in:
parent
3952cef556
commit
7841e3f6c0
@ -29,7 +29,7 @@ from __future__ import absolute_import
|
|||||||
|
|
||||||
import os
|
import os
|
||||||
import os.path
|
import os.path
|
||||||
import string
|
import string # pylint: disable=deprecated-module
|
||||||
|
|
||||||
import lxml.etree
|
import lxml.etree
|
||||||
import pkg_resources
|
import pkg_resources
|
||||||
@ -49,58 +49,118 @@ class StoragePoolException(qubes.exc.QubesException):
|
|||||||
class Volume(object):
|
class Volume(object):
|
||||||
''' Encapsulates all data about a volume for serialization to qubes.xml and
|
''' Encapsulates all data about a volume for serialization to qubes.xml and
|
||||||
libvirt config.
|
libvirt config.
|
||||||
|
|
||||||
|
|
||||||
|
Keep in mind!
|
||||||
|
volatile = not snap_on_start and not save_on_stop
|
||||||
|
snapshot = snap_on_start and not save_on_stop
|
||||||
|
origin = not snap_on_start and save_on_stop
|
||||||
|
origin_snapshot = snap_on_start and save_on_stop
|
||||||
'''
|
'''
|
||||||
|
|
||||||
devtype = 'disk'
|
devtype = 'disk'
|
||||||
domain = None
|
domain = None
|
||||||
path = None
|
path = None
|
||||||
rw = True
|
|
||||||
script = None
|
script = None
|
||||||
usage = 0
|
usage = 0
|
||||||
|
|
||||||
def __init__(self, name, pool, volume_type, vid=None, size=0,
|
def __init__(self, name, pool, vid, internal=False, removable=False,
|
||||||
removable=False, internal=False, **kwargs):
|
revisions_to_keep=0, rw=False, save_on_stop=False, size=0,
|
||||||
|
snap_on_start=False, source=None, **kwargs):
|
||||||
|
''' Initialize a volume.
|
||||||
|
|
||||||
|
:param str name: The domain name
|
||||||
|
:param str pool: The pool name
|
||||||
|
:param str vid: Volume identifier needs to be unique in pool
|
||||||
|
:param bool internal: If `True` volume is hidden when qvm-block ls
|
||||||
|
is used
|
||||||
|
:param bool removable: If `True` volume can be detached from vm at
|
||||||
|
run time
|
||||||
|
:param int revisions_to_keep: Amount of revisions to keep around
|
||||||
|
:param bool rw: If true volume will be mounted read-write
|
||||||
|
:param bool snap_on_start: Create a snapshot from source on start
|
||||||
|
:param bool save_on_stop: Write changes to disk in vm.stop()
|
||||||
|
:param str source: Vid of other volume in same pool
|
||||||
|
:param str/int size: Size of the volume
|
||||||
|
|
||||||
|
'''
|
||||||
|
|
||||||
super(Volume, self).__init__(**kwargs)
|
super(Volume, self).__init__(**kwargs)
|
||||||
|
|
||||||
self.name = str(name)
|
self.name = str(name)
|
||||||
self.pool = str(pool)
|
self.pool = str(pool)
|
||||||
self.vid = vid
|
|
||||||
self.size = size
|
|
||||||
self.volume_type = volume_type
|
|
||||||
self.removable = removable
|
|
||||||
self.internal = internal
|
self.internal = internal
|
||||||
|
self.removable = removable
|
||||||
|
self.revisions_to_keep = revisions_to_keep
|
||||||
|
self.rw = rw
|
||||||
|
self.save_on_stop = save_on_stop
|
||||||
|
self.size = int(size)
|
||||||
|
self.snap_on_start = snap_on_start
|
||||||
|
self.source = source
|
||||||
|
self.vid = vid
|
||||||
|
|
||||||
def __xml__(self):
|
def __eq__(self, other):
|
||||||
return lxml.etree.Element('volume', **self.config)
|
return other.pool == self.pool and other.vid == self.vid
|
||||||
|
|
||||||
@property
|
def __hash__(self):
|
||||||
def config(self):
|
return hash('%s:%s' % (self.pool, self.vid))
|
||||||
''' return config data for serialization to qubes.xml '''
|
|
||||||
return {'name': self.name,
|
def __neq__(self, other):
|
||||||
'pool': self.pool,
|
return not self.__eq__(other)
|
||||||
'volume_type': self.volume_type}
|
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return '{!r}'.format(self.pool + ':' + self.vid)
|
return '{!r}'.format(self.pool + ':' + self.vid)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return str(self.vid)
|
||||||
|
|
||||||
|
def __xml__(self):
|
||||||
|
config = _sanitize_config(self.config)
|
||||||
|
return lxml.etree.Element('volume', **config)
|
||||||
|
|
||||||
def block_device(self):
|
def block_device(self):
|
||||||
''' Return :py:class:`qubes.devices.BlockDevice` for serialization in
|
''' Return :py:class:`qubes.devices.BlockDevice` for serialization in
|
||||||
the libvirt XML template as <disk>.
|
the libvirt XML template as <disk>.
|
||||||
'''
|
'''
|
||||||
return qubes.devices.BlockDevice(self.path, self.name, self.script,
|
return qubes.devices.BlockDevice(self.path, self.name, self.script,
|
||||||
self.rw, self.domain, self.devtype)
|
self.rw, self.domain, self.devtype)
|
||||||
|
|
||||||
def __eq__(self, other):
|
@property
|
||||||
return other.pool == self.pool and other.vid == self.vid \
|
def revisions(self):
|
||||||
and other.volume_type == self.volume_type
|
''' Returns a `dict` containing revision identifiers and paths '''
|
||||||
|
msg = "{!s} has revisions not implemented".format(self.__class__)
|
||||||
|
raise NotImplementedError(msg)
|
||||||
|
|
||||||
def __neq__(self, other):
|
@property
|
||||||
return not self.__eq__(other)
|
def config(self):
|
||||||
|
''' return config data for serialization to qubes.xml '''
|
||||||
|
result = {'name': self.name, 'pool': self.pool, 'vid': self.vid, }
|
||||||
|
|
||||||
def __hash__(self):
|
if self.internal:
|
||||||
return hash('%s:%s %s' % (self.pool, self.vid, self.volume_type))
|
result['internal'] = self.internal
|
||||||
|
|
||||||
def __str__(self):
|
if self.removable:
|
||||||
return "{!s}:{!s}".format(self.pool, self.vid)
|
result['removable'] = self.removable
|
||||||
|
|
||||||
|
if self.revisions_to_keep:
|
||||||
|
result['revisions_to_keep'] = self.revisions_to_keep
|
||||||
|
|
||||||
|
if self.rw:
|
||||||
|
result['rw'] = self.rw
|
||||||
|
|
||||||
|
if self.save_on_stop:
|
||||||
|
result['save_on_stop'] = self.save_on_stop
|
||||||
|
|
||||||
|
if self.size:
|
||||||
|
result['size'] = self.size
|
||||||
|
|
||||||
|
if self.snap_on_start:
|
||||||
|
result['snap_on_start'] = self.snap_on_start
|
||||||
|
|
||||||
|
if self.source:
|
||||||
|
result['source'] = self.source
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
class Storage(object):
|
class Storage(object):
|
||||||
@ -119,6 +179,7 @@ class Storage(object):
|
|||||||
#: Additional drive (currently used only by HVM)
|
#: Additional drive (currently used only by HVM)
|
||||||
self.drive = None
|
self.drive = None
|
||||||
self.pools = {}
|
self.pools = {}
|
||||||
|
|
||||||
if hasattr(vm, 'volume_config'):
|
if hasattr(vm, 'volume_config'):
|
||||||
for name, conf in self.vm.volume_config.items():
|
for name, conf in self.vm.volume_config.items():
|
||||||
assert 'pool' in conf, "Pool missing in volume_config" % str(
|
assert 'pool' in conf, "Pool missing in volume_config" % str(
|
||||||
@ -187,7 +248,7 @@ class Storage(object):
|
|||||||
If :py:attr:`self.vm.kernel` is :py:obj:`None`, the this points inside
|
If :py:attr:`self.vm.kernel` is :py:obj:`None`, the this points inside
|
||||||
:py:attr:`self.vm.dir_path`
|
:py:attr:`self.vm.dir_path`
|
||||||
'''
|
'''
|
||||||
assert 'kernel' in self.vm.volumes, "VM has no kernel pool"
|
assert 'kernel' in self.vm.volumes, "VM has no kernel volume"
|
||||||
return self.vm.volumes['kernel'].kernels_dir
|
return self.vm.volumes['kernel'].kernels_dir
|
||||||
|
|
||||||
def get_disk_utilization(self):
|
def get_disk_utilization(self):
|
||||||
@ -248,11 +309,15 @@ class Storage(object):
|
|||||||
def rename(self, old_name, new_name):
|
def rename(self, old_name, new_name):
|
||||||
''' Notify the pools that the domain was renamed '''
|
''' Notify the pools that the domain was renamed '''
|
||||||
volumes = self.vm.volumes
|
volumes = self.vm.volumes
|
||||||
|
vm = self.vm
|
||||||
|
old_dir_path = os.path.join(os.path.dirname(vm.dir_path), old_name)
|
||||||
|
new_dir_path = os.path.join(os.path.dirname(vm.dir_path), new_name)
|
||||||
|
os.rename(old_dir_path, new_dir_path)
|
||||||
for name, volume in volumes.items():
|
for name, volume in volumes.items():
|
||||||
pool = self.get_pool(volume)
|
pool = self.get_pool(volume)
|
||||||
volumes[name] = pool.rename(volume, old_name, new_name)
|
volumes[name] = pool.rename(volume, old_name, new_name)
|
||||||
|
|
||||||
def verify_files(self):
|
def verify(self):
|
||||||
'''Verify that the storage is sane.
|
'''Verify that the storage is sane.
|
||||||
|
|
||||||
On success, returns normally. On failure, raises exception.
|
On success, returns normally. On failure, raises exception.
|
||||||
@ -264,6 +329,7 @@ class Storage(object):
|
|||||||
for volume in self.vm.volumes.values():
|
for volume in self.vm.volumes.values():
|
||||||
self.get_pool(volume).verify(volume)
|
self.get_pool(volume).verify(volume)
|
||||||
self.vm.fire_event('domain-verify-files')
|
self.vm.fire_event('domain-verify-files')
|
||||||
|
return True
|
||||||
|
|
||||||
def remove(self):
|
def remove(self):
|
||||||
''' Remove all the volumes.
|
''' Remove all the volumes.
|
||||||
@ -280,7 +346,8 @@ class Storage(object):
|
|||||||
def start(self):
|
def start(self):
|
||||||
''' Execute the start method on each pool '''
|
''' Execute the start method on each pool '''
|
||||||
for volume in self.vm.volumes.values():
|
for volume in self.vm.volumes.values():
|
||||||
self.get_pool(volume).start(volume)
|
pool = self.get_pool(volume)
|
||||||
|
volume = pool.start(volume)
|
||||||
|
|
||||||
def stop(self):
|
def stop(self):
|
||||||
''' Execute the start method on each pool '''
|
''' Execute the start method on each pool '''
|
||||||
@ -289,8 +356,12 @@ class Storage(object):
|
|||||||
|
|
||||||
def get_pool(self, volume):
|
def get_pool(self, volume):
|
||||||
''' Helper function '''
|
''' Helper function '''
|
||||||
assert isinstance(volume, Volume), "You need to pass a Volume"
|
assert isinstance(volume, (Volume, basestring)), \
|
||||||
return self.pools[volume.name]
|
"You need to pass a Volume or pool name as str"
|
||||||
|
if isinstance(volume, Volume):
|
||||||
|
return self.pools[volume.name]
|
||||||
|
else:
|
||||||
|
return self.vm.app.pools[volume]
|
||||||
|
|
||||||
def commit_template_changes(self):
|
def commit_template_changes(self):
|
||||||
''' Makes changes to an 'origin' volume persistent '''
|
''' Makes changes to an 'origin' volume persistent '''
|
||||||
@ -320,106 +391,129 @@ class Pool(object):
|
|||||||
|
|
||||||
3rd Parties providing own storage implementations will need to extend
|
3rd Parties providing own storage implementations will need to extend
|
||||||
this class.
|
this class.
|
||||||
'''
|
''' # pylint: disable=unused-argument
|
||||||
private_img_size = qubes.config.defaults['private_img_size']
|
private_img_size = qubes.config.defaults['private_img_size']
|
||||||
root_img_size = qubes.config.defaults['root_img_size']
|
root_img_size = qubes.config.defaults['root_img_size']
|
||||||
|
|
||||||
|
def __init__(self, name, revisions_to_keep=1, **kwargs):
|
||||||
|
super(Pool, self).__init__(**kwargs)
|
||||||
|
self.name = name
|
||||||
|
self.revisions_to_keep = revisions_to_keep
|
||||||
|
kwargs['name'] = self.name
|
||||||
|
|
||||||
def __eq__(self, other):
|
def __eq__(self, other):
|
||||||
return self.name == other.name
|
return self.name == other.name
|
||||||
|
|
||||||
|
|
||||||
def __neq__(self, other):
|
def __neq__(self, other):
|
||||||
return not self.__eq__(other)
|
return not self.__eq__(other)
|
||||||
|
|
||||||
def __init__(self, name, **kwargs):
|
|
||||||
super(Pool, self).__init__(**kwargs)
|
|
||||||
self.name = name
|
|
||||||
kwargs['name'] = self.name
|
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
def __xml__(self):
|
def __xml__(self):
|
||||||
return lxml.etree.Element('pool', **self.config)
|
config = _sanitize_config(self.config)
|
||||||
|
return lxml.etree.Element('pool', **config)
|
||||||
|
|
||||||
def create(self, volume, source_volume=None):
|
def create(self, volume):
|
||||||
''' Create the given volume on disk or copy from provided
|
''' Create the given volume on disk or copy from provided
|
||||||
`source_volume`.
|
`source_volume`.
|
||||||
'''
|
'''
|
||||||
raise NotImplementedError("Pool %s has create() not implemented" %
|
raise self._not_implemented("create")
|
||||||
self.name)
|
|
||||||
|
|
||||||
def commit_template_changes(self, volume):
|
def commit(self, volume): # pylint: disable=no-self-use
|
||||||
''' Update origin device '''
|
''' Write the snapshot to disk '''
|
||||||
raise NotImplementedError(
|
msg = "Got volume_type {!s} when expected 'snap'"
|
||||||
"Pool %s has commit_template_changes() not implemented" %
|
msg = msg.format(volume.volume_type)
|
||||||
self.name)
|
assert volume.volume_type == 'snap', msg
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def config(self):
|
def config(self):
|
||||||
''' Returns the pool config to be written to qubes.xml '''
|
''' Returns the pool config to be written to qubes.xml '''
|
||||||
raise NotImplementedError("Pool %s has config() not implemented" %
|
raise self._not_implemented("config")
|
||||||
self.name)
|
|
||||||
|
|
||||||
def clone(self, source, target):
|
def clone(self, source, target):
|
||||||
''' Clone volume '''
|
''' Clone volume '''
|
||||||
raise NotImplementedError("Pool %s has clone() not implemented" %
|
raise self._not_implemented("clone")
|
||||||
self.name)
|
|
||||||
|
|
||||||
def destroy(self):
|
def destroy(self):
|
||||||
''' Called when removing the pool. Use this for implementation specific
|
''' Called when removing the pool. Use this for implementation specific
|
||||||
clean up.
|
clean up.
|
||||||
'''
|
'''
|
||||||
raise NotImplementedError("Pool %s has destroy() not implemented" %
|
raise self._not_implemented("destroy")
|
||||||
self.name)
|
|
||||||
|
def export(self, volume):
|
||||||
|
''' Returns an object that can be `open()`. '''
|
||||||
|
raise self._not_implemented("export")
|
||||||
|
|
||||||
|
def import_volume(self, dst_pool, dst_volume, src_pool, src_volume):
|
||||||
|
''' Imports data to a volume in this pool '''
|
||||||
|
raise self._not_implemented("import_volume")
|
||||||
|
|
||||||
|
def init_volume(self, vm, volume_config):
|
||||||
|
''' Initialize a :py:class:`qubes.storage.Volume` from `volume_config`.
|
||||||
|
'''
|
||||||
|
raise self._not_implemented("init_volume")
|
||||||
|
|
||||||
|
def is_dirty(self, volume):
|
||||||
|
''' Return `True` if volume was not properly shutdown and commited '''
|
||||||
|
raise self._not_implemented("is_dirty")
|
||||||
|
|
||||||
def is_outdated(self, volume):
|
def is_outdated(self, volume):
|
||||||
raise NotImplementedError("Pool %s has is_outdated() not implemented" %
|
''' Returns `True` if the currently used `volume.source` of a snapshot
|
||||||
self.name)
|
volume is outdated.
|
||||||
|
'''
|
||||||
|
raise self._not_implemented("is_outdated")
|
||||||
|
|
||||||
|
def recover(self, volume):
|
||||||
|
''' Try to recover a :py:class:`Volume` or :py:class:`SnapVolume` '''
|
||||||
|
raise self._not_implemented("recover")
|
||||||
|
|
||||||
def remove(self, volume):
|
def remove(self, volume):
|
||||||
''' Remove volume'''
|
''' Remove volume'''
|
||||||
raise NotImplementedError("Pool %s has remove() not implemented" %
|
raise self._not_implemented("remove")
|
||||||
self.name)
|
|
||||||
|
|
||||||
def rename(self, volume, old_name, new_name):
|
def rename(self, volume, old_name, new_name):
|
||||||
''' Called when the domain changes its name '''
|
''' Called when the domain changes its name '''
|
||||||
raise NotImplementedError("Pool %s has rename() not implemented" %
|
raise self._not_implemented("rename")
|
||||||
self.name)
|
|
||||||
|
|
||||||
def start(self, volume):
|
def reset(self, volume):
|
||||||
''' Do what ever is needed on start '''
|
''' Drop and recreate volume without copying it's content from source.
|
||||||
raise NotImplementedError("Pool %s has start() not implemented" %
|
'''
|
||||||
self.name)
|
raise self._not_implemented("reset")
|
||||||
|
|
||||||
|
def revert(self, volume, revision=None):
|
||||||
|
''' Revert volume to previous revision '''
|
||||||
|
raise self._not_implemented("revert")
|
||||||
|
|
||||||
def setup(self):
|
def setup(self):
|
||||||
''' Called when adding a pool to the system. Use this for implementation
|
''' Called when adding a pool to the system. Use this for implementation
|
||||||
specific set up.
|
specific set up.
|
||||||
'''
|
'''
|
||||||
raise NotImplementedError("Pool %s has setup() not implemented" %
|
raise self._not_implemented("setup")
|
||||||
self.name)
|
|
||||||
|
|
||||||
def stop(self, volume):
|
def start(self, volume): # pylint: disable=no-self-use
|
||||||
|
''' Do what ever is needed on start '''
|
||||||
|
raise self._not_implemented("start")
|
||||||
|
|
||||||
|
def stop(self, volume): # pylint: disable=no-self-use
|
||||||
''' Do what ever is needed on stop'''
|
''' Do what ever is needed on stop'''
|
||||||
raise NotImplementedError("Pool %s has stop() not implemented" %
|
|
||||||
self.name)
|
|
||||||
|
|
||||||
def init_volume(self, vm, volume_config):
|
|
||||||
''' Initialize a :py:class:`qubes.storage.Volume` from `volume_config`.
|
|
||||||
'''
|
|
||||||
raise NotImplementedError("Pool %s has init_volume() not implemented" %
|
|
||||||
self.name)
|
|
||||||
|
|
||||||
def verify(self, volume):
|
def verify(self, volume):
|
||||||
''' Verifies the volume. '''
|
''' Verifies the volume. '''
|
||||||
raise NotImplementedError("Pool %s has verify() not implemented" %
|
raise self._not_implemented("verify")
|
||||||
self.name)
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def volumes(self):
|
def volumes(self):
|
||||||
''' Return a list of volumes managed by this pool '''
|
''' Return a list of volumes managed by this pool '''
|
||||||
raise NotImplementedError("Pool %s has volumes() not implemented" %
|
raise self._not_implemented("volumes")
|
||||||
self.name)
|
|
||||||
|
def _not_implemented(self, method_name):
|
||||||
|
''' Helper for emitting helpful `NotImplementedError` exceptions '''
|
||||||
|
msg = "Pool driver {!s} has {!s}() not implemented"
|
||||||
|
msg = msg.format(str(self.__class__.__name__), method_name)
|
||||||
|
return NotImplementedError(msg)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def pool_drivers():
|
def pool_drivers():
|
||||||
|
Loading…
Reference in New Issue
Block a user