storage/lvm: fix Volume() instance init when physical volume is unavailable

First, cache objects created with init_volume - this is the only place
where we have full volume configuration (including snap_on_start and
save_on_stop properties).
But also implement get_volume method, to get a volume instance for given
volume id. Such volume instance may be incomplete (other attributes are
available only in owning domain configuration), but it will be enough
for basic operations - like cheching and changing its size, cloning
etc.
Listing volumes still use list of physically present volumes.

This makes it possible to start qubesd service, without physical
presence of some storage devices. Starting VMs using such storage would
still fail, of course.

Fixes QubesOS/qubes-issues#2960
This commit is contained in:
Marek Marczykowski-Górecki 2017-08-12 22:44:03 +02:00
parent 9ad85a3dff
commit 6e5fe58128
No known key found for this signature in database
GPG Key ID: 063938BA42CFA724

View File

@ -61,6 +61,8 @@ class ThinPool(qubes.storage.Pool):
self._pool_id = "{!s}/{!s}".format(volume_group, thin_pool) self._pool_id = "{!s}/{!s}".format(volume_group, thin_pool)
self.log = logging.getLogger('qube.storage.lvm.%s' % self._pool_id) self.log = logging.getLogger('qube.storage.lvm.%s' % self._pool_id)
self._volume_objects_cache = {}
@property @property
def config(self): def config(self):
return { return {
@ -91,11 +93,27 @@ class ThinPool(qubes.storage.Pool):
volume_config['volume_group'] = self.volume_group volume_config['volume_group'] = self.volume_group
volume_config['pool'] = self volume_config['pool'] = self
return ThinVolume(**volume_config) volume = ThinVolume(**volume_config)
self._volume_objects_cache[volume_config['vid']] = volume
return volume
def setup(self): def setup(self):
pass # TODO Should we create a non existing pool? pass # TODO Should we create a non existing pool?
def get_volume(self, vid):
''' Return a volume with given vid'''
if vid in self._volume_objects_cache:
return self._volume_objects_cache[vid]
config = {
'pool': self,
'vid': vid,
'name': vid,
'volume_group': self.volume_group,
}
# don't cache this object, as it doesn't carry full configuration
return ThinVolume(**config)
def list_volumes(self): def list_volumes(self):
''' Return a list of volumes managed by this pool ''' ''' Return a list of volumes managed by this pool '''
volumes = [] volumes = []
@ -288,6 +306,8 @@ class ThinVolume(qubes.storage.Volume):
cmd = ['remove', self.vid] cmd = ['remove', self.vid]
qubes_lvm(cmd, self.log) qubes_lvm(cmd, self.log)
reset_cache() reset_cache()
# pylint: disable=protected-access
self.pool._volume_objects_cache.pop(self.vid, None)
def export(self): def export(self):
''' Returns an object that can be `open()`. ''' ''' Returns an object that can be `open()`. '''