2015-01-19 18:03:23 +01:00
|
|
|
#!/usr/bin/python2 -O
|
|
|
|
# vim: fileencoding=utf-8
|
|
|
|
|
2015-01-16 15:33:03 +01:00
|
|
|
#
|
2015-01-19 18:03:23 +01:00
|
|
|
# The Qubes OS Project, https://www.qubes-os.org/
|
2015-01-16 15:33:03 +01:00
|
|
|
#
|
2015-01-19 18:03:23 +01:00
|
|
|
# Copyright (C) 2015 Joanna Rutkowska <joanna@invisiblethingslab.com>
|
|
|
|
# Copyright (C) 2013-2015 Marek Marczykowski-Górecki
|
|
|
|
# <marmarek@invisiblethingslab.com>
|
2015-01-16 15:33:03 +01:00
|
|
|
# Copyright (C) 2015 Wojtek Porczyk <woju@invisiblethingslab.com>
|
|
|
|
#
|
2015-01-19 18:03:23 +01:00
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
2015-01-16 15:33:03 +01:00
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
2015-01-19 18:03:23 +01:00
|
|
|
# You should have received a copy of the GNU General Public License along
|
|
|
|
# with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
2015-01-16 15:33:03 +01:00
|
|
|
#
|
|
|
|
|
|
|
|
from __future__ import absolute_import
|
|
|
|
|
|
|
|
import os
|
|
|
|
import os.path
|
2015-01-19 18:14:15 +01:00
|
|
|
import re
|
2015-01-16 15:33:03 +01:00
|
|
|
import subprocess
|
|
|
|
|
2016-04-15 20:15:18 +02:00
|
|
|
from qubes.storage import Pool, StoragePoolException, Volume
|
2015-01-16 15:33:03 +01:00
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
BLKSIZE = 512
|
2015-01-16 15:33:03 +01:00
|
|
|
|
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
class XenPool(Pool):
|
|
|
|
''' File based 'original' disk implementation '''
|
2016-04-15 16:01:27 +02:00
|
|
|
driver = 'xen'
|
2015-01-16 15:33:03 +01:00
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
def __init__(self, name=None, dir_path=None):
|
|
|
|
super(XenPool, self).__init__(name=name)
|
2016-04-15 20:15:18 +02:00
|
|
|
assert dir_path, "No pool dir_path specified"
|
|
|
|
self.dir_path = os.path.normpath(dir_path)
|
2016-03-03 01:05:23 +01:00
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
create_dir_if_not_exists(self.dir_path)
|
2016-04-15 20:15:18 +02:00
|
|
|
appvms_path = os.path.join(self.dir_path, 'appvms')
|
2016-04-15 20:40:53 +02:00
|
|
|
create_dir_if_not_exists(appvms_path)
|
2016-04-15 20:15:18 +02:00
|
|
|
vm_templates_path = os.path.join(self.dir_path, 'vm-templates')
|
2016-04-15 20:40:53 +02:00
|
|
|
create_dir_if_not_exists(vm_templates_path)
|
|
|
|
|
2016-04-15 15:16:25 +02:00
|
|
|
def clone(self, source, target):
|
|
|
|
''' Clones the volume if the `source.pool` if the source is a
|
|
|
|
:py:class:`XenVolume`.
|
|
|
|
'''
|
|
|
|
if issubclass(XenVolume, source.__class__):
|
|
|
|
raise StoragePoolException('Volumes %s and %s use different pools'
|
|
|
|
% (source.__class__, target.__class__))
|
|
|
|
|
|
|
|
if source.volume_type not in ['origin', 'read-write']:
|
|
|
|
return target
|
|
|
|
|
|
|
|
copy_file(source.vid, target.vid)
|
|
|
|
return target
|
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
def create(self, volume, source_volume=None):
|
|
|
|
_type = volume.volume_type
|
|
|
|
size = volume.size
|
|
|
|
if _type == 'origin':
|
|
|
|
create_sparse_file(volume.path_origin, size)
|
|
|
|
create_sparse_file(volume.path_cow, size)
|
|
|
|
elif _type in ['read-write'] and source_volume:
|
|
|
|
copy_file(source_volume.path, volume.path)
|
|
|
|
elif _type in ['read-write', 'volatile']:
|
|
|
|
create_sparse_file(volume.path, size)
|
2016-03-03 01:05:23 +01:00
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
return volume
|
2015-01-16 15:33:03 +01:00
|
|
|
|
2016-04-15 16:01:27 +02:00
|
|
|
@property
|
|
|
|
def config(self):
|
|
|
|
return {
|
|
|
|
'name': self.name,
|
|
|
|
'dir_path': self.dir_path,
|
|
|
|
'driver': XenPool.driver,
|
|
|
|
}
|
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
def resize(self, volume, size):
|
|
|
|
''' Expands volume, throws
|
|
|
|
:py:class:`qubst.storage.StoragePoolException` if given size is
|
|
|
|
less than current_size
|
|
|
|
'''
|
|
|
|
_type = volume.volume_type
|
|
|
|
if _type not in ['origin', 'read-write', 'volatile']:
|
|
|
|
raise StoragePoolException('Can not resize a %s volume %s' %
|
|
|
|
(_type, volume.vid))
|
|
|
|
|
|
|
|
if size <= volume.size:
|
|
|
|
raise StoragePoolException(
|
|
|
|
'For your own safety, shrinking of %s is'
|
|
|
|
' disabled. If you really know what you'
|
|
|
|
' are doing, use `truncate` on %s manually.' %
|
|
|
|
(volume.name, volume.vid))
|
|
|
|
|
|
|
|
if _type == 'origin':
|
|
|
|
path = volume.path_origin
|
|
|
|
elif _type in ['read-write', 'volatile']:
|
|
|
|
path = volume.path
|
|
|
|
|
|
|
|
with open(path, 'a+b') as fd:
|
|
|
|
fd.truncate(size)
|
2015-01-16 15:33:03 +01:00
|
|
|
|
2016-04-05 20:04:20 +02:00
|
|
|
self._resize_loop_device(path)
|
|
|
|
|
2016-04-15 17:10:55 +02:00
|
|
|
def remove(self, volume):
|
|
|
|
if volume.volume_type in ['read-write', 'volatile']:
|
|
|
|
_remove_if_exists(volume.vid)
|
|
|
|
elif volume.volume_type == 'origin':
|
|
|
|
_remove_if_exists(volume.vid)
|
|
|
|
_remove_if_exists(volume.path_cow)
|
|
|
|
|
2016-04-05 20:04:20 +02:00
|
|
|
def _resize_loop_device(self, path):
|
2015-01-16 15:33:03 +01:00
|
|
|
# find loop device if any
|
2016-04-05 20:04:20 +02:00
|
|
|
p = subprocess.Popen(
|
|
|
|
['sudo', 'losetup', '--associated', path],
|
|
|
|
stdout=subprocess.PIPE)
|
2015-01-16 15:33:03 +01:00
|
|
|
result = p.communicate()
|
|
|
|
|
|
|
|
m = re.match(r'^(/dev/loop\d+):\s', result[0])
|
|
|
|
if m is not None:
|
|
|
|
loop_dev = m.group(1)
|
|
|
|
|
|
|
|
# resize loop device
|
2016-04-15 20:15:18 +02:00
|
|
|
subprocess.check_call(['sudo', 'losetup', '--set-capacity',
|
|
|
|
loop_dev])
|
2015-01-16 15:33:03 +01:00
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
def commit_template_changes(self, volume):
|
|
|
|
if volume.volume_type != 'origin':
|
|
|
|
return volume
|
2015-01-16 15:33:03 +01:00
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
if os.path.exists(volume.path_cow):
|
|
|
|
os.rename(volume.path_cow, volume.path_cow + '.old')
|
2015-01-16 15:33:03 +01:00
|
|
|
|
|
|
|
old_umask = os.umask(002)
|
2016-04-15 20:40:53 +02:00
|
|
|
with open(volume.path_cow, 'w') as f_cow:
|
|
|
|
f_cow.truncate(volume.size)
|
2015-01-16 15:33:03 +01:00
|
|
|
os.umask(old_umask)
|
2016-04-01 13:44:29 +02:00
|
|
|
return volume
|
|
|
|
|
|
|
|
def start(self, volume):
|
|
|
|
if volume.volume_type == 'volatile':
|
|
|
|
self._reset_volume(volume)
|
2016-04-15 20:40:53 +02:00
|
|
|
if volume.volume_type in ['origin', 'snapshot']:
|
|
|
|
_check_path(volume.path_origin)
|
|
|
|
_check_path(volume.path_cow)
|
|
|
|
else:
|
|
|
|
_check_path(volume.path)
|
|
|
|
|
2016-04-01 13:44:29 +02:00
|
|
|
return volume
|
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
def stop(self, volume):
|
|
|
|
pass
|
|
|
|
|
2016-04-01 13:44:29 +02:00
|
|
|
def _reset_volume(self, volume):
|
|
|
|
''' Remove and recreate a volatile volume '''
|
|
|
|
assert volume.volume_type == 'volatile', "Not a volatile volume"
|
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
assert volume.size
|
2016-04-01 13:44:29 +02:00
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
_remove_if_exists(volume)
|
2016-04-01 13:44:29 +02:00
|
|
|
|
|
|
|
with open(volume.path, "w") as f_volatile:
|
|
|
|
f_volatile.truncate(volume.size)
|
|
|
|
return volume
|
2016-03-03 01:05:23 +01:00
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
def target_dir(self, vm):
|
2016-04-15 20:15:18 +02:00
|
|
|
""" 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
|
|
|
|
"""
|
|
|
|
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)
|
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
def init_volume(self, vm, volume_config):
|
2016-04-14 19:00:52 +02:00
|
|
|
assert 'volume_type' in volume_config, "Volume type missing " \
|
|
|
|
+ str(volume_config)
|
|
|
|
volume_type = volume_config['volume_type']
|
|
|
|
known_types = {
|
|
|
|
'read-write': ReadWriteFile,
|
|
|
|
'read-only': ReadOnlyFile,
|
|
|
|
'origin': OriginFile,
|
|
|
|
'snapshot': SnapshotFile,
|
|
|
|
'volatile': VolatileFile,
|
|
|
|
}
|
|
|
|
if volume_type not in known_types:
|
|
|
|
raise StoragePoolException("Unknown volume type " + volume_type)
|
2016-04-15 20:40:48 +02:00
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
if volume_type in ['snapshot', 'read-only']:
|
|
|
|
origin_pool = vm.app.get_pool(volume_config['pool'])
|
|
|
|
assert isinstance(origin_pool,
|
|
|
|
XenPool), 'Origin volume not a xen volume'
|
|
|
|
volume_config['target_dir'] = origin_pool.target_dir(vm.template)
|
|
|
|
name = volume_config['name']
|
|
|
|
volume_config['size'] = vm.template.volume_config[name]['size']
|
|
|
|
else:
|
|
|
|
volume_config['target_dir'] = self.target_dir(vm)
|
2016-04-15 20:40:48 +02:00
|
|
|
|
2016-04-14 19:00:52 +02:00
|
|
|
return known_types[volume_type](**volume_config)
|
|
|
|
|
2016-04-15 19:59:40 +02:00
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
class XenVolume(Volume):
|
2016-04-15 15:16:25 +02:00
|
|
|
''' Parent class for the xen volumes implementation which expects a
|
|
|
|
`target_dir` param on initialization.
|
|
|
|
'''
|
2016-04-15 20:40:53 +02:00
|
|
|
|
|
|
|
def __init__(self, target_dir, **kwargs):
|
|
|
|
self.target_dir = target_dir
|
|
|
|
assert self.target_dir, "target_dir not specified"
|
|
|
|
super(XenVolume, self).__init__(**kwargs)
|
|
|
|
|
2016-04-15 19:59:40 +02:00
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
class SizeMixIn(XenVolume):
|
|
|
|
''' A mix in which expects a `size` param to be > 0 on initialization and
|
|
|
|
provides a usage property wrapper.
|
|
|
|
'''
|
2016-04-15 15:16:25 +02:00
|
|
|
|
|
|
|
def __init__(self, size=0, **kwargs):
|
|
|
|
super(SizeMixIn, self).__init__(size=int(size), **kwargs)
|
|
|
|
assert size, 'Empty size provided'
|
|
|
|
assert size > 0, 'Size for volume ' + kwargs['name'] + ' is <=0'
|
2016-04-15 19:59:40 +02:00
|
|
|
|
|
|
|
@property
|
2016-04-15 20:40:53 +02:00
|
|
|
def usage(self):
|
|
|
|
''' Returns the actualy used space '''
|
|
|
|
return get_disk_usage(self.vid)
|
|
|
|
|
2016-04-15 19:57:23 +02:00
|
|
|
@property
|
|
|
|
def config(self):
|
|
|
|
''' return config data for serialization to qubes.xml '''
|
|
|
|
return {'name': self.name,
|
|
|
|
'pool': self.pool,
|
|
|
|
'size': str(self.size),
|
|
|
|
'volume_type': self.volume_type}
|
2016-04-15 19:59:40 +02:00
|
|
|
|
|
|
|
|
|
|
|
class ReadWriteFile(SizeMixIn):
|
2016-04-15 20:40:53 +02:00
|
|
|
# :pylint: disable=missing-docstring
|
2016-04-15 19:59:40 +02:00
|
|
|
def __init__(self, **kwargs):
|
|
|
|
super(ReadWriteFile, self).__init__(**kwargs)
|
|
|
|
self.path = os.path.join(self.target_dir, self.name + '.img')
|
|
|
|
self.vid = self.path
|
|
|
|
|
|
|
|
|
2016-04-15 15:16:25 +02:00
|
|
|
class ReadOnlyFile(XenVolume):
|
2016-04-15 20:40:53 +02:00
|
|
|
# :pylint: disable=missing-docstring
|
|
|
|
usage = 0
|
2016-04-15 19:59:40 +02:00
|
|
|
|
2016-04-15 15:16:25 +02:00
|
|
|
def __init__(self, size=0, **kwargs):
|
2016-04-15 20:40:53 +02:00
|
|
|
# :pylint: disable=unused-argument
|
2016-04-15 15:16:25 +02:00
|
|
|
super(ReadOnlyFile, self).__init__(size=int(size), **kwargs)
|
2016-04-15 19:59:40 +02:00
|
|
|
self.path = self.vid
|
|
|
|
|
|
|
|
|
|
|
|
class OriginFile(SizeMixIn):
|
2016-04-15 20:40:53 +02:00
|
|
|
# :pylint: disable=missing-docstring
|
2016-04-15 19:59:40 +02:00
|
|
|
script = 'block-origin'
|
|
|
|
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
super(OriginFile, self).__init__(**kwargs)
|
2016-04-15 20:40:53 +02:00
|
|
|
self.path_origin = os.path.join(self.target_dir, self.name + '.img')
|
2016-04-15 19:59:40 +02:00
|
|
|
self.path_cow = os.path.join(self.target_dir, self.name + '-cow.img')
|
2016-04-15 20:40:53 +02:00
|
|
|
self.path = '%s:%s' % (self.path_origin, self.path_cow)
|
|
|
|
self.vid = self.path_origin
|
2016-04-15 19:59:40 +02:00
|
|
|
|
|
|
|
def commit(self):
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
@property
|
2016-04-15 20:40:53 +02:00
|
|
|
def usage(self):
|
|
|
|
result = 0
|
|
|
|
if os.path.exists(self.path_origin):
|
|
|
|
result += get_disk_usage(self.path_origin)
|
|
|
|
if os.path.exists(self.path_cow):
|
|
|
|
result += get_disk_usage(self.path_cow)
|
|
|
|
return result
|
2016-04-15 19:59:40 +02:00
|
|
|
|
|
|
|
|
2016-04-15 15:16:25 +02:00
|
|
|
class SnapshotFile(XenVolume):
|
2016-04-15 20:40:53 +02:00
|
|
|
# :pylint: disable=missing-docstring
|
2016-04-15 19:59:40 +02:00
|
|
|
script = 'block-snapshot'
|
|
|
|
rw = False
|
2016-04-15 20:40:53 +02:00
|
|
|
usage = 0
|
2016-04-15 19:59:40 +02:00
|
|
|
|
2016-04-15 15:16:25 +02:00
|
|
|
def __init__(self, name=None, size=None, **kwargs):
|
2016-04-15 20:40:53 +02:00
|
|
|
assert size
|
2016-04-15 15:16:25 +02:00
|
|
|
super(SnapshotFile, self).__init__(name=name, size=int(size), **kwargs)
|
|
|
|
self.path_origin = os.path.join(self.target_dir, name + '.img')
|
|
|
|
self.path_cow = os.path.join(self.target_dir, name + '-cow.img')
|
2016-04-15 20:40:53 +02:00
|
|
|
self.path = '%s:%s' % (self.path_origin, self.path_cow)
|
|
|
|
self.vid = self.path_origin
|
2016-04-15 19:59:40 +02:00
|
|
|
|
|
|
|
@property
|
|
|
|
def created(self):
|
2016-04-15 20:40:53 +02:00
|
|
|
return os.path.exists(self.path_origin) and os.path.exists(
|
|
|
|
self.path_cow)
|
2016-04-15 19:59:40 +02:00
|
|
|
|
|
|
|
|
|
|
|
class VolatileFile(SizeMixIn):
|
2016-04-15 20:40:53 +02:00
|
|
|
# :pylint: disable=missing-docstring
|
2016-04-15 19:59:40 +02:00
|
|
|
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
super(VolatileFile, self).__init__(**kwargs)
|
|
|
|
self.path = os.path.join(self.target_dir, 'volatile.img')
|
|
|
|
self.vid = self.path
|
|
|
|
|
|
|
|
|
2016-04-15 20:40:53 +02:00
|
|
|
def create_sparse_file(path, size):
|
|
|
|
''' Create an empty sparse file '''
|
2016-04-15 19:59:40 +02:00
|
|
|
if os.path.exists(path):
|
|
|
|
raise IOError("Volume %s already exists", path)
|
2016-04-15 20:40:53 +02:00
|
|
|
parent_dir = os.path.dirname(path)
|
|
|
|
if not os.path.exists(parent_dir):
|
|
|
|
os.makedirs(parent_dir)
|
2016-04-15 19:59:40 +02:00
|
|
|
with open(path, 'a+b') as fh:
|
|
|
|
fh.truncate(size)
|
2016-04-15 20:40:53 +02:00
|
|
|
|
|
|
|
|
|
|
|
def get_disk_usage_one(st):
|
|
|
|
'''Extract disk usage of one inode from its stat_result struct.
|
|
|
|
|
|
|
|
If known, get real disk usage, as written to device by filesystem, not
|
|
|
|
logical file size. Those values may be different for sparse files.
|
|
|
|
|
|
|
|
:param os.stat_result st: stat result
|
|
|
|
:returns: disk usage
|
|
|
|
'''
|
|
|
|
try:
|
|
|
|
return st.st_blocks * BLKSIZE
|
|
|
|
except AttributeError:
|
|
|
|
return st.st_size
|
|
|
|
|
|
|
|
|
|
|
|
def get_disk_usage(path):
|
|
|
|
'''Get real disk usage of given path (file or directory).
|
|
|
|
|
|
|
|
When *path* points to directory, then it is evaluated recursively.
|
|
|
|
|
|
|
|
This function tries estiate real disk usage. See documentation of
|
|
|
|
:py:func:`get_disk_usage_one`.
|
|
|
|
|
|
|
|
:param str path: path to evaluate
|
|
|
|
:returns: disk usage
|
|
|
|
'''
|
|
|
|
try:
|
|
|
|
st = os.lstat(path)
|
|
|
|
except OSError:
|
|
|
|
return 0
|
|
|
|
|
|
|
|
ret = get_disk_usage_one(st)
|
|
|
|
|
|
|
|
# if path is not a directory, this is skipped
|
|
|
|
for dirpath, dirnames, filenames in os.walk(path):
|
|
|
|
for name in dirnames + filenames:
|
|
|
|
ret += get_disk_usage_one(os.lstat(os.path.join(dirpath, name)))
|
|
|
|
|
|
|
|
return ret
|
|
|
|
|
|
|
|
|
|
|
|
def create_dir_if_not_exists(path):
|
|
|
|
""" Check if a directory exists in if not create it.
|
|
|
|
|
|
|
|
This method does not create any parent directories.
|
|
|
|
"""
|
|
|
|
if not os.path.exists(path):
|
|
|
|
os.mkdir(path)
|
|
|
|
|
|
|
|
|
|
|
|
def copy_file(source, destination):
|
|
|
|
'''Effective file copy, preserving sparse files etc.
|
|
|
|
'''
|
|
|
|
# TODO: Windows support
|
|
|
|
# We prefer to use Linux's cp, because it nicely handles sparse files
|
|
|
|
assert os.path.exists(source), \
|
|
|
|
"Missing the source %s to copy from" % source
|
|
|
|
assert not os.path.exists(destination), \
|
|
|
|
"Destination %s already exists" % destination
|
|
|
|
|
|
|
|
parent_dir = os.path.dirname(destination)
|
|
|
|
if not os.path.exists(parent_dir):
|
|
|
|
os.makedirs(parent_dir)
|
|
|
|
|
|
|
|
try:
|
|
|
|
subprocess.check_call(['cp', '--reflink=auto', source, destination])
|
|
|
|
except subprocess.CalledProcessError:
|
|
|
|
raise IOError('Error while copying {!r} to {!r}'.format(source,
|
|
|
|
destination))
|
|
|
|
|
|
|
|
|
|
|
|
def _remove_if_exists(volume):
|
|
|
|
if os.path.exists(volume.path):
|
|
|
|
os.remove(volume.path)
|
|
|
|
|
|
|
|
|
|
|
|
def _check_path(path):
|
|
|
|
''' Raise an StoragePoolException if ``path`` does not exist'''
|
|
|
|
if not os.path.exists(path):
|
|
|
|
raise StoragePoolException('Missing image file: %s' % path)
|