Fix lvm size/usage
- Uses a size_cache, because it's faster than doing a call for each volume
This commit is contained in:
parent
0561dfce60
commit
1ca3c00797
@ -32,6 +32,8 @@ class ThinPool(qubes.storage.Pool):
|
|||||||
''' LVM Thin based pool implementation
|
''' LVM Thin based pool implementation
|
||||||
''' # pylint: disable=protected-access
|
''' # pylint: disable=protected-access
|
||||||
|
|
||||||
|
size_cache = None
|
||||||
|
|
||||||
driver = 'lvm_thin'
|
driver = 'lvm_thin'
|
||||||
|
|
||||||
def __init__(self, volume_group, thin_pool, revisions_to_keep=1, **kwargs):
|
def __init__(self, volume_group, thin_pool, revisions_to_keep=1, **kwargs):
|
||||||
@ -90,6 +92,7 @@ class ThinPool(qubes.storage.Pool):
|
|||||||
str(volume.size)
|
str(volume.size)
|
||||||
]
|
]
|
||||||
qubes_lvm(cmd, self.log)
|
qubes_lvm(cmd, self.log)
|
||||||
|
reset_cache()
|
||||||
return volume
|
return volume
|
||||||
|
|
||||||
def destroy(self):
|
def destroy(self):
|
||||||
@ -146,6 +149,7 @@ class ThinPool(qubes.storage.Pool):
|
|||||||
dst.write(tmp)
|
dst.write(tmp)
|
||||||
p.stdin.close()
|
p.stdin.close()
|
||||||
p.wait()
|
p.wait()
|
||||||
|
reset_cache()
|
||||||
return dst_volume
|
return dst_volume
|
||||||
|
|
||||||
def is_dirty(self, volume):
|
def is_dirty(self, volume):
|
||||||
@ -161,6 +165,7 @@ class ThinPool(qubes.storage.Pool):
|
|||||||
|
|
||||||
cmd = ['remove', volume.vid]
|
cmd = ['remove', volume.vid]
|
||||||
qubes_lvm(cmd, self.log)
|
qubes_lvm(cmd, self.log)
|
||||||
|
reset_cache()
|
||||||
|
|
||||||
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 '''
|
||||||
@ -179,6 +184,7 @@ class ThinPool(qubes.storage.Pool):
|
|||||||
if not volume._is_volatile:
|
if not volume._is_volatile:
|
||||||
volume._vid_snap = volume.vid + '-snap'
|
volume._vid_snap = volume.vid + '-snap'
|
||||||
return volume
|
return volume
|
||||||
|
reset_cache()
|
||||||
|
|
||||||
def revert(self, volume, revision=None):
|
def revert(self, volume, revision=None):
|
||||||
old_path = volume.path + '-back'
|
old_path = volume.path + '-back'
|
||||||
@ -190,6 +196,7 @@ class ThinPool(qubes.storage.Pool):
|
|||||||
qubes_lvm(cmd, self.log)
|
qubes_lvm(cmd, self.log)
|
||||||
cmd = ['clone', volume.vid + '-back', volume.vid]
|
cmd = ['clone', volume.vid + '-back', volume.vid]
|
||||||
qubes_lvm(cmd, self.log)
|
qubes_lvm(cmd, self.log)
|
||||||
|
reset_cache()
|
||||||
return volume
|
return volume
|
||||||
|
|
||||||
def resize(self, volume, size):
|
def resize(self, volume, size):
|
||||||
@ -210,7 +217,7 @@ class ThinPool(qubes.storage.Pool):
|
|||||||
|
|
||||||
cmd = ['extend', volume.vid, str(size)]
|
cmd = ['extend', volume.vid, str(size)]
|
||||||
qubes_lvm(cmd, self.log)
|
qubes_lvm(cmd, self.log)
|
||||||
volume.size += size
|
reset_cache()
|
||||||
|
|
||||||
def _reset(self, volume):
|
def _reset(self, volume):
|
||||||
try:
|
try:
|
||||||
@ -232,6 +239,7 @@ class ThinPool(qubes.storage.Pool):
|
|||||||
if not self.is_dirty(volume):
|
if not self.is_dirty(volume):
|
||||||
self._snapshot(volume)
|
self._snapshot(volume)
|
||||||
|
|
||||||
|
reset_cache
|
||||||
return volume
|
return volume
|
||||||
|
|
||||||
def stop(self, volume):
|
def stop(self, volume):
|
||||||
@ -244,6 +252,7 @@ class ThinPool(qubes.storage.Pool):
|
|||||||
else:
|
else:
|
||||||
cmd = ['remove', volume._vid_snap]
|
cmd = ['remove', volume._vid_snap]
|
||||||
qubes_lvm(cmd, self.log)
|
qubes_lvm(cmd, self.log)
|
||||||
|
reset_cache()
|
||||||
return volume
|
return volume
|
||||||
|
|
||||||
def _snapshot(self, volume):
|
def _snapshot(self, volume):
|
||||||
@ -301,10 +310,41 @@ class ThinPool(qubes.storage.Pool):
|
|||||||
str(volume.size)]
|
str(volume.size)]
|
||||||
qubes_lvm(cmd, self.log)
|
qubes_lvm(cmd, self.log)
|
||||||
|
|
||||||
|
|
||||||
|
def init_cache(log=logging.getLogger('qube.storage.lvm')):
|
||||||
|
cmd = ['sudo', 'lvs', '--noheadings', '-o',
|
||||||
|
'vg_name,name,lv_size,data_percent', '--units', 'b', '--separator',
|
||||||
|
',']
|
||||||
|
p = subprocess.Popen(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
|
||||||
|
out, err = p.communicate()
|
||||||
|
return_code = p.returncode
|
||||||
|
if return_code == 0 and err:
|
||||||
|
log.warning(err)
|
||||||
|
elif return_code != 0:
|
||||||
|
raise qubes.storage.StoragePoolException(err)
|
||||||
|
|
||||||
|
result = {}
|
||||||
|
|
||||||
|
for l in out.splitlines():
|
||||||
|
l = l.strip()
|
||||||
|
pool_name, name, size, usage_percent = l.split(',', 3)
|
||||||
|
if '' in [pool_name, name, size, usage_percent]:
|
||||||
|
continue
|
||||||
|
name = pool_name + "/" + name
|
||||||
|
size = int(size[:-1])
|
||||||
|
usage = int(size / 100 * float(usage_percent))
|
||||||
|
result[name] = {'size':size, 'usage': usage}
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
size_cache = init_cache()
|
||||||
|
|
||||||
class ThinVolume(qubes.storage.Volume):
|
class ThinVolume(qubes.storage.Volume):
|
||||||
''' Default LVM thin volume implementation
|
''' Default LVM thin volume implementation
|
||||||
''' # pylint: disable=too-few-public-methods
|
''' # pylint: disable=too-few-public-methods
|
||||||
|
|
||||||
|
|
||||||
def __init__(self, volume_group, **kwargs):
|
def __init__(self, volume_group, **kwargs):
|
||||||
self.volume_group = volume_group
|
self.volume_group = volume_group
|
||||||
super(ThinVolume, self).__init__(**kwargs)
|
super(ThinVolume, self).__init__(**kwargs)
|
||||||
@ -347,6 +387,22 @@ class ThinVolume(qubes.storage.Volume):
|
|||||||
def _is_volatile(self):
|
def _is_volatile(self):
|
||||||
return not self.snap_on_start and not self.save_on_stop
|
return not self.snap_on_start and not self.save_on_stop
|
||||||
|
|
||||||
|
@property
|
||||||
|
def size(self):
|
||||||
|
try:
|
||||||
|
return qubes.storage.lvm.size_cache[self.vid]['size']
|
||||||
|
except KeyError:
|
||||||
|
return self._size
|
||||||
|
|
||||||
|
@property
|
||||||
|
def usage(self): # lvm thin usage always returns at least the same usage as
|
||||||
|
# the parent
|
||||||
|
try:
|
||||||
|
return qubes.storage.lvm.size_cache[self.vid]['usage']
|
||||||
|
except KeyError:
|
||||||
|
return 0
|
||||||
|
|
||||||
|
|
||||||
def pool_exists(pool_id):
|
def pool_exists(pool_id):
|
||||||
''' Return true if pool exists '''
|
''' Return true if pool exists '''
|
||||||
cmd = ['pool', pool_id]
|
cmd = ['pool', pool_id]
|
||||||
@ -369,3 +425,7 @@ def qubes_lvm(cmd, log=logging.getLogger('qube.storage.lvm')):
|
|||||||
assert err, "Command exited unsuccessful, but printed nothing to stderr"
|
assert err, "Command exited unsuccessful, but printed nothing to stderr"
|
||||||
raise qubes.storage.StoragePoolException(err)
|
raise qubes.storage.StoragePoolException(err)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def reset_cache(log=logging.getLogger('qube.storage.lvm')):
|
||||||
|
qubes.storage.lvm.size_cache = init_cache
|
||||||
|
Loading…
Reference in New Issue
Block a user