Replace XenStorage with XenPool
This commit is contained in:
parent
32255a7916
commit
4d4b846ce8
@ -33,41 +33,32 @@ import subprocess
|
|||||||
|
|
||||||
import qubes
|
import qubes
|
||||||
import qubes.config
|
import qubes.config
|
||||||
import qubes.storage
|
|
||||||
import qubes.vm.templatevm
|
import qubes.vm.templatevm
|
||||||
from qubes.devices import BlockDevice
|
from qubes.storage import Pool, StoragePoolException, Volume
|
||||||
|
|
||||||
|
|
||||||
class XenStorage(qubes.storage.Storage):
|
class XenPool(Pool):
|
||||||
'''Class for VM storage of Xen VMs.
|
|
||||||
'''
|
|
||||||
|
|
||||||
root_dev = 'xvda'
|
root_dev = 'xvda'
|
||||||
private_dev = 'xvdb'
|
private_dev = 'xvdb'
|
||||||
volatile_dev = 'xvdc'
|
volatile_dev = 'xvdc'
|
||||||
modules_dev = 'xvdd'
|
|
||||||
|
|
||||||
def __init__(self, vm, vmdir, **kwargs):
|
def __init__(self, vm=None, name=None, dir_path=None):
|
||||||
""" Instantiate the storage.
|
super(XenPool, self).__init__(vm=vm, name=name)
|
||||||
|
assert dir_path, "No pool dir_path specified"
|
||||||
Args:
|
self.dir_path = os.path.normpath(dir_path)
|
||||||
vm: a QubesVM
|
|
||||||
vmdir: the root directory of the pool
|
|
||||||
"""
|
|
||||||
assert vm is not None
|
|
||||||
assert vmdir is not None
|
|
||||||
|
|
||||||
super(XenStorage, self).__init__(vm, **kwargs)
|
|
||||||
|
|
||||||
self.vmdir = vmdir
|
|
||||||
|
|
||||||
|
self.create_dir_if_not_exists(self.dir_path)
|
||||||
|
appvms_path = os.path.join(self.dir_path, 'appvms')
|
||||||
|
self.create_dir_if_not_exists(appvms_path)
|
||||||
|
vm_templates_path = os.path.join(self.dir_path, 'vm-templates')
|
||||||
|
self.create_dir_if_not_exists(vm_templates_path)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def private_img(self):
|
def private_img(self):
|
||||||
'''Path to the private image'''
|
'''Path to the private image'''
|
||||||
return self.abspath(qubes.config.vm_files['private_img'])
|
return self.abspath(qubes.config.vm_files['private_img'])
|
||||||
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def root_img(self):
|
def root_img(self):
|
||||||
'''Path to the root image'''
|
'''Path to the root image'''
|
||||||
@ -75,7 +66,6 @@ class XenStorage(qubes.storage.Storage):
|
|||||||
if hasattr(self.vm, 'template') and self.vm.template \
|
if hasattr(self.vm, 'template') and self.vm.template \
|
||||||
else self.abspath(qubes.config.vm_files['root_img'])
|
else self.abspath(qubes.config.vm_files['root_img'])
|
||||||
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def rootcow_img(self):
|
def rootcow_img(self):
|
||||||
'''Path to the root COW image'''
|
'''Path to the root COW image'''
|
||||||
@ -85,17 +75,11 @@ class XenStorage(qubes.storage.Storage):
|
|||||||
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def volatile_img(self):
|
def volatile_img(self):
|
||||||
'''Path to the volatile image'''
|
'''Path to the volatile image'''
|
||||||
return self.abspath(qubes.config.vm_files['volatile_img'])
|
return self.abspath(qubes.config.vm_files['volatile_img'])
|
||||||
|
|
||||||
def format_disk_dev(self, path, name, script=None, rw=True, devtype='disk',
|
|
||||||
domain=None):
|
|
||||||
|
|
||||||
return BlockDevice(path, name, script, rw, domain, devtype)
|
|
||||||
|
|
||||||
def root_dev_config(self):
|
def root_dev_config(self):
|
||||||
dev_name = 'root'
|
dev_name = 'root'
|
||||||
if isinstance(self.vm, qubes.vm.templatevm.TemplateVM):
|
if isinstance(self.vm, qubes.vm.templatevm.TemplateVM):
|
||||||
@ -123,27 +107,27 @@ class XenStorage(qubes.storage.Storage):
|
|||||||
# any other template-based VM - two device-mapper layers: one
|
# any other template-based VM - two device-mapper layers: one
|
||||||
# in dom0 (here) from root+root-cow, and another one from
|
# in dom0 (here) from root+root-cow, and another one from
|
||||||
# this+volatile.img
|
# this+volatile.img
|
||||||
return self.format_disk_dev(
|
path = '{root}:{template_rootcow}'.format(
|
||||||
'{root}:{template_rootcow}'.format(
|
root=self.root_img,
|
||||||
root=self.root_img,
|
template_rootcow=self.vm.template.storage.rootcow_img)
|
||||||
template_rootcow=self.vm.template.storage.rootcow_img),
|
return self.format_disk_dev(path=path,
|
||||||
self.root_dev,
|
vdev=self.root_dev,
|
||||||
script='block-snapshot',
|
script='block-snapshot',
|
||||||
rw=False)
|
rw=False)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
# standalone qube
|
# standalone qube
|
||||||
return self.format_disk_dev(self.root_img, dev_name)
|
return self.format_disk_dev(self.root_img, dev_name)
|
||||||
|
|
||||||
|
|
||||||
def private_dev_config(self):
|
def private_dev_config(self):
|
||||||
return self.format_disk_dev(self.private_img, 'private')
|
return self.format_disk_dev(self.private_img, 'private')
|
||||||
|
|
||||||
def volatile_dev_config(self):
|
def volatile_dev_config(self):
|
||||||
return self.format_disk_dev(self.volatile_img, 'volatile')
|
return self.format_disk_dev(self.volatile_img, 'volatile')
|
||||||
|
|
||||||
|
|
||||||
def create_on_disk_private_img(self, source_template=None):
|
def create_on_disk_private_img(self, source_template=None):
|
||||||
|
if not os.path.exists(self.target_dir):
|
||||||
|
os.makedirs(self.target_dir)
|
||||||
if source_template is None:
|
if source_template is None:
|
||||||
f_private = open(self.private_img, 'a+b')
|
f_private = open(self.private_img, 'a+b')
|
||||||
f_private.truncate(self.private_img_size)
|
f_private.truncate(self.private_img_size)
|
||||||
@ -152,11 +136,11 @@ class XenStorage(qubes.storage.Storage):
|
|||||||
else:
|
else:
|
||||||
self.vm.log.info("Copying the template's private image: {}".format(
|
self.vm.log.info("Copying the template's private image: {}".format(
|
||||||
source_template.storage.private_img))
|
source_template.storage.private_img))
|
||||||
self._copy_file(source_template.storage.private_img,
|
self._copy_file(source_template.storage.private_img, self.private_img)
|
||||||
self.private_img)
|
|
||||||
|
|
||||||
|
|
||||||
def create_on_disk_root_img(self, source_template=None):
|
def create_on_disk_root_img(self, source_template=None):
|
||||||
|
if not os.path.exists(self.target_dir):
|
||||||
|
os.makedirs(self.target_dir)
|
||||||
if source_template is None:
|
if source_template is None:
|
||||||
fd = open(self.root_img, 'a+b')
|
fd = open(self.root_img, 'a+b')
|
||||||
fd.truncate(self.root_img_size)
|
fd.truncate(self.root_img_size)
|
||||||
@ -164,11 +148,11 @@ class XenStorage(qubes.storage.Storage):
|
|||||||
|
|
||||||
elif self.vm.updateable:
|
elif self.vm.updateable:
|
||||||
# if not updateable, just use template's disk
|
# if not updateable, just use template's disk
|
||||||
self.vm.log.info("--> Copying the template's root image: {}".format(
|
self.vm.log.info(
|
||||||
source_template.storage.root_img))
|
"--> Copying the template's root image: {}".format(
|
||||||
|
source_template.storage.root_img))
|
||||||
self._copy_file(source_template.storage.root_img, self.root_img)
|
self._copy_file(source_template.storage.root_img, self.root_img)
|
||||||
|
|
||||||
|
|
||||||
def resize_private_img(self, size):
|
def resize_private_img(self, size):
|
||||||
fd = open(self.private_img, 'a+b')
|
fd = open(self.private_img, 'a+b')
|
||||||
fd.truncate(size)
|
fd.truncate(size)
|
||||||
@ -185,9 +169,8 @@ class XenStorage(qubes.storage.Storage):
|
|||||||
loop_dev = m.group(1)
|
loop_dev = m.group(1)
|
||||||
|
|
||||||
# resize loop device
|
# resize loop device
|
||||||
subprocess.check_call(
|
subprocess.check_call(['sudo', 'losetup', '--set-capacity',
|
||||||
['sudo', 'losetup', '--set-capacity', loop_dev])
|
loop_dev])
|
||||||
|
|
||||||
|
|
||||||
def commit_template_changes(self):
|
def commit_template_changes(self):
|
||||||
assert isinstance(self.vm, qubes.vm.templatevm.TemplateVM)
|
assert isinstance(self.vm, qubes.vm.templatevm.TemplateVM)
|
||||||
@ -206,7 +189,6 @@ class XenStorage(qubes.storage.Storage):
|
|||||||
f_root.close()
|
f_root.close()
|
||||||
os.umask(old_umask)
|
os.umask(old_umask)
|
||||||
|
|
||||||
|
|
||||||
def reset_volatile_storage(self):
|
def reset_volatile_storage(self):
|
||||||
try:
|
try:
|
||||||
# no template set, in any way (Standalone VM, Template VM)
|
# no template set, in any way (Standalone VM, Template VM)
|
||||||
@ -241,15 +223,14 @@ class XenStorage(qubes.storage.Storage):
|
|||||||
f_volatile.truncate(f_root.tell())
|
f_volatile.truncate(f_root.tell())
|
||||||
f_volatile.close()
|
f_volatile.close()
|
||||||
f_root.close()
|
f_root.close()
|
||||||
return # XXX why is that? super() does not run
|
return # XXX why is that? super() does not run
|
||||||
except AttributeError: # self.vm.template
|
except AttributeError: # self.vm.template
|
||||||
pass
|
pass
|
||||||
|
|
||||||
super(XenStorage, self).reset_volatile_storage()
|
super(XenPool, self).reset_volatile_storage()
|
||||||
|
|
||||||
|
|
||||||
def prepare_for_vm_startup(self):
|
def prepare_for_vm_startup(self):
|
||||||
super(XenStorage, self).prepare_for_vm_startup()
|
super(XenPool, self).prepare_for_vm_startup()
|
||||||
|
|
||||||
if self.drive is not None:
|
if self.drive is not None:
|
||||||
# pylint: disable=unused-variable
|
# pylint: disable=unused-variable
|
||||||
@ -260,14 +241,44 @@ class XenStorage(qubes.storage.Storage):
|
|||||||
drive_vm = self.vm.app.domains[drive_domain]
|
drive_vm = self.vm.app.domains[drive_domain]
|
||||||
|
|
||||||
if not drive_vm.is_running():
|
if not drive_vm.is_running():
|
||||||
raise qubes.exc.QubesVMNotRunningError(drive_vm,
|
raise qubes.exc.QubesVMNotRunningError(
|
||||||
'VM {!r} holding {!r} isn\'t running'.format(
|
drive_vm, 'VM {!r} holding {!r} isn\'t running'.format(
|
||||||
drive_domain, drive_path))
|
drive_domain, drive_path))
|
||||||
|
|
||||||
if self.rootcow_img and not os.path.exists(self.rootcow_img):
|
if self.rootcow_img and not os.path.exists(self.rootcow_img):
|
||||||
self.commit_template_changes()
|
self.commit_template_changes()
|
||||||
|
|
||||||
class XenPool(qubes.storage.Pool):
|
# XXX there is also a class attribute on the domain classes which does
|
||||||
def get_storage(self):
|
# exactly that -- which one should prevail?
|
||||||
""" Returns an instantiated ``XenStorage``. """
|
@property
|
||||||
return XenStorage(self.vm, vmdir=self.vmdir)
|
def target_dir(self):
|
||||||
|
""" Returns the path to vmdir depending on the type of the VM.
|
||||||
|
|
||||||
|
The default QubesOS file storage saves the vm images in three
|
||||||
|
different directories depending on the ``QubesVM`` type:
|
||||||
|
|
||||||
|
* ``appvms`` for ``QubesAppVm`` or ``QubesHvm``
|
||||||
|
* ``vm-templates`` for ``QubesTemplateVm`` or ``QubesTemplateHvm``
|
||||||
|
|
||||||
|
Args:
|
||||||
|
vm: a QubesVM
|
||||||
|
pool_dir: the root directory of the pool
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
string (str) absolute path to the directory where the vm files
|
||||||
|
are stored
|
||||||
|
"""
|
||||||
|
vm = self.vm
|
||||||
|
if vm.is_template():
|
||||||
|
subdir = 'vm-templates'
|
||||||
|
elif vm.is_disposablevm():
|
||||||
|
subdir = 'appvms'
|
||||||
|
return os.path.join(self.dir_path, subdir,
|
||||||
|
vm.template.name + '-dvm')
|
||||||
|
else:
|
||||||
|
subdir = 'appvms'
|
||||||
|
|
||||||
|
return os.path.join(self.dir_path, subdir, vm.name)
|
||||||
|
|
||||||
|
def abspath(self, file_name):
|
||||||
|
return os.path.join(self.target_dir, file_name)
|
||||||
|
Loading…
Reference in New Issue
Block a user