storage/lvm: use temporary volume for data import
Do not write directly to main volume, instead create temporary volume and only commit it to the main one when operation is finished. This solve multiple problems: - import operation can be aborted, without data loss - importing new data over existing volume will not leave traces of previous content - especially when importing smaller volume to bigger one - import operation can be reverted - it create separate revision, similar to start/stop - easier to prevent qube from starting during import operation - template still can be used when importing new version QubesOS/qubes-issues#2256
This commit is contained in:
parent
aea0de35ad
commit
2b80f0c044
@ -159,7 +159,7 @@ class ThinPool(qubes.storage.Pool):
|
|||||||
continue
|
continue
|
||||||
if vol_info['pool_lv'] != self.thin_pool:
|
if vol_info['pool_lv'] != self.thin_pool:
|
||||||
continue
|
continue
|
||||||
if vid.endswith('-snap'):
|
if vid.endswith('-snap') or vid.endswith('-import'):
|
||||||
# implementation detail volume
|
# implementation detail volume
|
||||||
continue
|
continue
|
||||||
if vid.endswith('-back'):
|
if vid.endswith('-back'):
|
||||||
@ -249,6 +249,8 @@ class ThinVolume(qubes.storage.Volume):
|
|||||||
|
|
||||||
if self.snap_on_start or self.save_on_stop:
|
if self.snap_on_start or self.save_on_stop:
|
||||||
self._vid_snap = self.vid + '-snap'
|
self._vid_snap = self.vid + '-snap'
|
||||||
|
if self.save_on_stop:
|
||||||
|
self._vid_import = self.vid + '-import'
|
||||||
|
|
||||||
self._size = size
|
self._size = size
|
||||||
|
|
||||||
@ -409,6 +411,13 @@ class ThinVolume(qubes.storage.Volume):
|
|||||||
except AttributeError:
|
except AttributeError:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
if os.path.exists('/dev/' + self._vid_import):
|
||||||
|
cmd = ['remove', self._vid_import]
|
||||||
|
qubes_lvm(cmd, self.log)
|
||||||
|
except AttributeError:
|
||||||
|
pass
|
||||||
|
|
||||||
self._remove_revisions(self.revisions.keys())
|
self._remove_revisions(self.revisions.keys())
|
||||||
if not os.path.exists(self.path):
|
if not os.path.exists(self.path):
|
||||||
return
|
return
|
||||||
@ -434,36 +443,74 @@ class ThinVolume(qubes.storage.Volume):
|
|||||||
raise qubes.storage.StoragePoolException(
|
raise qubes.storage.StoragePoolException(
|
||||||
'Cannot import to dirty volume {} -'
|
'Cannot import to dirty volume {} -'
|
||||||
' start and stop a qube to cleanup'.format(self.vid))
|
' start and stop a qube to cleanup'.format(self.vid))
|
||||||
|
self.abort_if_import_in_progress()
|
||||||
# HACK: neat trick to speed up testing if you have same physical thin
|
# HACK: neat trick to speed up testing if you have same physical thin
|
||||||
# pool assigned to two qubes-pools i.e: qubes_dom0 and test-lvm
|
# pool assigned to two qubes-pools i.e: qubes_dom0 and test-lvm
|
||||||
# pylint: disable=line-too-long
|
# pylint: disable=line-too-long
|
||||||
if isinstance(src_volume.pool, ThinPool) and \
|
if isinstance(src_volume.pool, ThinPool) and \
|
||||||
src_volume.pool.thin_pool == self.pool.thin_pool: # NOQA
|
src_volume.pool.thin_pool == self.pool.thin_pool: # NOQA
|
||||||
self._commit(src_volume.path, keep=True)
|
self._commit(src_volume.path[len('/dev/'):], keep=True)
|
||||||
else:
|
else:
|
||||||
cmd = ['create', self.pool._pool_id, self._vid_snap.split('/')[1],
|
cmd = ['create',
|
||||||
|
self.pool._pool_id, # pylint: disable=protected-access
|
||||||
|
self._vid_import.split('/')[1],
|
||||||
str(src_volume.size)]
|
str(src_volume.size)]
|
||||||
qubes_lvm(cmd)
|
qubes_lvm(cmd, self.log)
|
||||||
src_path = src_volume.export()
|
src_path = src_volume.export()
|
||||||
cmd = ['dd', 'if=' + src_path, 'of=/dev/' + self._vid_snap,
|
cmd = ['dd', 'if=' + src_path, 'of=/dev/' + self._vid_import,
|
||||||
'conv=sparse']
|
'conv=sparse', 'status=none']
|
||||||
|
if not os.access('/dev/' + self._vid_import, os.W_OK) or \
|
||||||
|
not os.access(src_path, os.R_OK):
|
||||||
|
cmd.insert(0, 'sudo')
|
||||||
|
|
||||||
p = yield from asyncio.create_subprocess_exec(*cmd)
|
p = yield from asyncio.create_subprocess_exec(*cmd)
|
||||||
yield from p.wait()
|
yield from p.wait()
|
||||||
if p.returncode != 0:
|
if p.returncode != 0:
|
||||||
cmd = ['remove', self._vid_snap]
|
cmd = ['remove', self._vid_import]
|
||||||
qubes_lvm(cmd)
|
qubes_lvm(cmd, self.log)
|
||||||
raise qubes.storage.StoragePoolException(
|
raise qubes.storage.StoragePoolException(
|
||||||
'Failed to import volume {!r}, dd exit code: {}'.format(
|
'Failed to import volume {!r}, dd exit code: {}'.format(
|
||||||
src_volume, p.returncode))
|
src_volume, p.returncode))
|
||||||
self._commit()
|
self._commit(self._vid_import)
|
||||||
|
|
||||||
return self
|
return self
|
||||||
|
|
||||||
def import_data(self):
|
def import_data(self):
|
||||||
''' Returns an object that can be `open()`. '''
|
''' Returns an object that can be `open()`. '''
|
||||||
devpath = '/dev/' + self.vid
|
if self.is_dirty():
|
||||||
|
raise qubes.storage.StoragePoolException(
|
||||||
|
'Cannot import data to dirty volume {}, stop the qube first'.
|
||||||
|
format(self.vid))
|
||||||
|
self.abort_if_import_in_progress()
|
||||||
|
# pylint: disable=protected-access
|
||||||
|
cmd = ['create', self.pool._pool_id, self._vid_import.split('/')[1],
|
||||||
|
str(self.size)]
|
||||||
|
qubes_lvm(cmd, self.log)
|
||||||
|
reset_cache()
|
||||||
|
devpath = '/dev/' + self._vid_import
|
||||||
return devpath
|
return devpath
|
||||||
|
|
||||||
|
def import_data_end(self, success):
|
||||||
|
'''Either commit imported data, or discard temporary volume'''
|
||||||
|
if not os.path.exists('/dev/' + self._vid_import):
|
||||||
|
raise qubes.storage.StoragePoolException(
|
||||||
|
'No import operation in progress on {}'.format(self.vid))
|
||||||
|
if success:
|
||||||
|
self._commit(self._vid_import)
|
||||||
|
else:
|
||||||
|
cmd = ['remove', self._vid_import]
|
||||||
|
qubes_lvm(cmd, self.log)
|
||||||
|
|
||||||
|
def abort_if_import_in_progress(self):
|
||||||
|
try:
|
||||||
|
devpath = '/dev/' + self._vid_import
|
||||||
|
if os.path.exists(devpath):
|
||||||
|
raise qubes.storage.StoragePoolException(
|
||||||
|
'Import operation in progress on {}'.format(self.vid))
|
||||||
|
except AttributeError: # self._vid_import
|
||||||
|
# no vid_import - import definitely not in progress
|
||||||
|
pass
|
||||||
|
|
||||||
def is_dirty(self):
|
def is_dirty(self):
|
||||||
if self.save_on_stop:
|
if self.save_on_stop:
|
||||||
return os.path.exists('/dev/' + self._vid_snap)
|
return os.path.exists('/dev/' + self._vid_snap)
|
||||||
@ -482,6 +529,7 @@ class ThinVolume(qubes.storage.Volume):
|
|||||||
raise qubes.storage.StoragePoolException(
|
raise qubes.storage.StoragePoolException(
|
||||||
'Cannot revert dirty volume {}, stop the qube first'.format(
|
'Cannot revert dirty volume {}, stop the qube first'.format(
|
||||||
self.vid))
|
self.vid))
|
||||||
|
self.abort_if_import_in_progress()
|
||||||
if revision is None:
|
if revision is None:
|
||||||
revision = \
|
revision = \
|
||||||
max(self.revisions.items(), key=_revision_sort_key)[0]
|
max(self.revisions.items(), key=_revision_sort_key)[0]
|
||||||
@ -520,6 +568,10 @@ class ThinVolume(qubes.storage.Volume):
|
|||||||
if self.is_dirty():
|
if self.is_dirty():
|
||||||
cmd = ['extend', self._vid_snap, str(size)]
|
cmd = ['extend', self._vid_snap, str(size)]
|
||||||
qubes_lvm(cmd, self.log)
|
qubes_lvm(cmd, self.log)
|
||||||
|
elif hasattr(self, '_vid_import') and \
|
||||||
|
os.path.exists('/dev/' + self._vid_import):
|
||||||
|
cmd = ['extend', self._vid_import, str(size)]
|
||||||
|
qubes_lvm(cmd, self.log)
|
||||||
elif self.save_on_stop or not self.snap_on_start:
|
elif self.save_on_stop or not self.snap_on_start:
|
||||||
cmd = ['extend', self._vid_current, str(size)]
|
cmd = ['extend', self._vid_current, str(size)]
|
||||||
qubes_lvm(cmd, self.log)
|
qubes_lvm(cmd, self.log)
|
||||||
@ -538,8 +590,8 @@ class ThinVolume(qubes.storage.Volume):
|
|||||||
cmd = ['clone', self.source.path, self._vid_snap]
|
cmd = ['clone', self.source.path, self._vid_snap]
|
||||||
qubes_lvm(cmd, self.log)
|
qubes_lvm(cmd, self.log)
|
||||||
|
|
||||||
|
|
||||||
def start(self):
|
def start(self):
|
||||||
|
self.abort_if_import_in_progress()
|
||||||
try:
|
try:
|
||||||
if self.snap_on_start or self.save_on_stop:
|
if self.snap_on_start or self.save_on_stop:
|
||||||
if not self.save_on_stop or not self.is_dirty():
|
if not self.save_on_stop or not self.is_dirty():
|
||||||
|
@ -649,6 +649,65 @@ class TC_00_ThinPool(ThinPoolBase):
|
|||||||
|
|
||||||
volume.remove()
|
volume.remove()
|
||||||
|
|
||||||
|
def test_030_import_data(self):
|
||||||
|
''' Test volume import'''
|
||||||
|
config = {
|
||||||
|
'name': 'root',
|
||||||
|
'pool': self.pool.name,
|
||||||
|
'save_on_stop': True,
|
||||||
|
'rw': True,
|
||||||
|
'revisions_to_keep': 2,
|
||||||
|
'size': qubes.config.defaults['root_img_size'],
|
||||||
|
}
|
||||||
|
vm = qubes.tests.storage.TestVM(self)
|
||||||
|
volume = self.app.get_pool(self.pool.name).init_volume(vm, config)
|
||||||
|
volume.create()
|
||||||
|
current_uuid = self._get_lv_uuid(volume.path)
|
||||||
|
self.assertFalse(volume.is_dirty())
|
||||||
|
import_path = volume.import_data()
|
||||||
|
import_uuid = self._get_lv_uuid(import_path)
|
||||||
|
self.assertNotEqual(current_uuid, import_uuid)
|
||||||
|
# success - commit data
|
||||||
|
volume.import_data_end(True)
|
||||||
|
new_current_uuid = self._get_lv_uuid(volume.path)
|
||||||
|
self.assertEqual(new_current_uuid, import_uuid)
|
||||||
|
revisions = volume.revisions
|
||||||
|
self.assertEqual(len(revisions), 1)
|
||||||
|
revision = revisions.popitem()[0]
|
||||||
|
self.assertEqual(current_uuid,
|
||||||
|
self._get_lv_uuid(volume.vid + '-' + revision))
|
||||||
|
self.assertFalse(os.path.exists(import_path), import_path)
|
||||||
|
|
||||||
|
volume.remove()
|
||||||
|
|
||||||
|
def test_031_import_data_fail(self):
|
||||||
|
''' Test volume import'''
|
||||||
|
config = {
|
||||||
|
'name': 'root',
|
||||||
|
'pool': self.pool.name,
|
||||||
|
'save_on_stop': True,
|
||||||
|
'rw': True,
|
||||||
|
'revisions_to_keep': 2,
|
||||||
|
'size': qubes.config.defaults['root_img_size'],
|
||||||
|
}
|
||||||
|
vm = qubes.tests.storage.TestVM(self)
|
||||||
|
volume = self.app.get_pool(self.pool.name).init_volume(vm, config)
|
||||||
|
volume.create()
|
||||||
|
current_uuid = self._get_lv_uuid(volume.path)
|
||||||
|
self.assertFalse(volume.is_dirty())
|
||||||
|
import_path = volume.import_data()
|
||||||
|
import_uuid = self._get_lv_uuid(import_path)
|
||||||
|
self.assertNotEqual(current_uuid, import_uuid)
|
||||||
|
# fail - discard data
|
||||||
|
volume.import_data_end(False)
|
||||||
|
new_current_uuid = self._get_lv_uuid(volume.path)
|
||||||
|
self.assertEqual(new_current_uuid, current_uuid)
|
||||||
|
revisions = volume.revisions
|
||||||
|
self.assertEqual(len(revisions), 0)
|
||||||
|
self.assertFalse(os.path.exists(import_path), import_path)
|
||||||
|
|
||||||
|
volume.remove()
|
||||||
|
|
||||||
|
|
||||||
@skipUnlessLvmPoolExists
|
@skipUnlessLvmPoolExists
|
||||||
class TC_01_ThinPool(ThinPoolBase, qubes.tests.SystemTestCase):
|
class TC_01_ThinPool(ThinPoolBase, qubes.tests.SystemTestCase):
|
||||||
|
Loading…
Reference in New Issue
Block a user