From 04a215fb83e770458c84220ad5df83fec7b4712a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marta=20Marczykowska-G=C3=B3recka?= Date: Thu, 8 Aug 2019 14:10:19 +0200 Subject: [PATCH 1/2] Add metadata info to LVM AdminAPI Added usage_details method to Pool class (returns a dictionary with detailed information on pool usage) and LVM implementation that returns metadata info. Needed for QubesOS/qubes-issues#5053 --- qubes/api/admin.py | 5 +++++ qubes/storage/__init__.py | 11 +++++++++++ qubes/storage/lvm.py | 32 ++++++++++++++++++++++++++++---- qubes/tests/api_admin.py | 7 ++++--- 4 files changed, 48 insertions(+), 7 deletions(-) diff --git a/qubes/api/admin.py b/qubes/api/admin.py index ea6e5b0e..e997cd4a 100644 --- a/qubes/api/admin.py +++ b/qubes/api/admin.py @@ -630,6 +630,11 @@ class QubesAdminAPI(qubes.api.AbstractQubesAPI): if pool_usage is not None: other_info += 'usage={}\n'.format(pool_usage) + pool_details = pool.usage_details + for name in pool_details: + if name not in ['data_size', 'data_usage']: + other_info += '{}={}\n'.format(name, pool_details[name]) + try: included_in = pool.included_in(self.app) if included_in: diff --git a/qubes/storage/__init__.py b/qubes/storage/__init__.py index 83451e22..b8bd79d9 100644 --- a/qubes/storage/__init__.py +++ b/qubes/storage/__init__.py @@ -795,6 +795,17 @@ class Pool: def usage(self): ''' Space used in the pool in bytes, or None if unknown ''' + @property + def usage_details(self): + """Detailed information about pool usage as a dictionary + Contains data_usage for usage in bytes and data_size for pool + size; other implementations may add more implementation-specific + detail""" + return { + 'data_usage': self.usage, + 'data_size': self.size + } + def _not_implemented(self, method_name): ''' Helper for emitting helpful `NotImplementedError` exceptions ''' msg = "Pool driver {!s} has {!s}() not implemented" diff --git a/qubes/storage/lvm.py b/qubes/storage/lvm.py index ad02f7a8..1c502d5e 100644 --- a/qubes/storage/lvm.py +++ b/qubes/storage/lvm.py @@ -195,10 +195,28 @@ class ThinPool(qubes.storage.Pool): except KeyError: return 0 + @property + def usage_details(self): + result = {} + result['data_size'] = self.size + result['data_usage'] = self.usage + + try: + metadata_size = qubes.storage.lvm.size_cache[ + self.volume_group + '/' + self.thin_pool]['metadata_size'] + metadata_usage = qubes.storage.lvm.size_cache[ + self.volume_group + '/' + self.thin_pool]['metadata_usage'] + except KeyError: + metadata_size = None + metadata_usage = None + result['metadata_size'] = metadata_size + result['metadata_usage'] = metadata_usage + + return result _init_cache_cmd = ['lvs', '--noheadings', '-o', - 'vg_name,pool_lv,name,lv_size,data_percent,lv_attr,origin', - '--units', 'b', '--separator', ';'] + 'vg_name,pool_lv,name,lv_size,data_percent,lv_attr,origin,lv_metadata_size,' + 'metadata_percent', '--units', 'b', '--separator', ';'] def _parse_lvm_cache(lvm_output): result = {} @@ -206,14 +224,20 @@ def _parse_lvm_cache(lvm_output): for line in lvm_output.splitlines(): line = line.decode().strip() pool_name, pool_lv, name, size, usage_percent, attr, \ - origin = line.split(';', 6) + origin, metadata_size, metadata_percent = line.split(';', 8) if '' in [pool_name, name, size, usage_percent]: continue name = pool_name + "/" + name size = int(size[:-1]) # Remove 'B' suffix usage = int(size / 100 * float(usage_percent)) + if metadata_size: + metadata_size = int(metadata_size[:-1]) + metadata_usage = int(metadata_size / 100 * float(metadata_percent)) + else: + metadata_usage = None result[name] = {'size': size, 'usage': usage, 'pool_lv': pool_lv, - 'attr': attr, 'origin': origin} + 'attr': attr, 'origin': origin, 'metadata_size': metadata_size, + 'metadata_usage': metadata_usage} return result diff --git a/qubes/tests/api_admin.py b/qubes/tests/api_admin.py index 76aea013..5db6225b 100644 --- a/qubes/tests/api_admin.py +++ b/qubes/tests/api_admin.py @@ -582,20 +582,21 @@ class TC_00_VMs(AdminAPITestCase): 'pool1': unittest.mock.Mock(config={ 'param1': 'value1', 'param2': 'value2'}, usage=102400, - size=204800) + size=204800, + usage_details={'metadata_size': 500}) } self.app.pools['pool1'].included_in.return_value = None value = self.call_mgmt_func(b'admin.pool.Info', b'dom0', b'pool1') self.assertEqual(value, - 'param1=value1\nparam2=value2\nsize=204800\nusage=102400\n') + 'param1=value1\nparam2=value2\nsize=204800\nusage=102400\nmetadata_size=500\n') self.assertFalse(self.app.save.called) def test_151_pool_info_unsupported_size(self): self.app.pools = { 'pool1': unittest.mock.Mock(config={ 'param1': 'value1', 'param2': 'value2'}, - size=None, usage=None), + size=None, usage=None, usage_details={}), } self.app.pools['pool1'].included_in.return_value = None value = self.call_mgmt_func(b'admin.pool.Info', b'dom0', b'pool1') From 2f6497e48da0ac795c5e211b2ed8975b5e937502 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marta=20Marczykowska-G=C3=B3recka?= Date: Tue, 22 Oct 2019 21:42:36 +0200 Subject: [PATCH 2/2] Added admin.pool.UsageDetails API method admin.pool.UsageDetails reports the usage data, unlike admin.pool.Info, which should report the config/unchangeable data. At the moment admin.Pool.Info still reports usage, to maintain compatibility, but once all relevant tools are updated, it should just return configuration data. --- Makefile | 1 + qubes/api/admin.py | 26 +++++++++++++++++++++----- qubes/storage/lvm.py | 4 ++-- qubes/tests/api_admin.py | 22 +++++++++++++++++++--- qubes/tests/storage_lvm.py | 32 ++++++++++++++++++++++++++++++++ 5 files changed, 75 insertions(+), 10 deletions(-) diff --git a/Makefile b/Makefile index 1fbe3699..e5538477 100644 --- a/Makefile +++ b/Makefile @@ -21,6 +21,7 @@ ADMIN_API_METHODS_SIMPLE = \ admin.label.Remove \ admin.pool.Add \ admin.pool.Info \ + admin.pool.UsageDetails \ admin.pool.List \ admin.pool.ListDrivers \ admin.pool.Remove \ diff --git a/qubes/api/admin.py b/qubes/api/admin.py index e997cd4a..f9a9c35c 100644 --- a/qubes/api/admin.py +++ b/qubes/api/admin.py @@ -622,6 +622,7 @@ class QubesAdminAPI(qubes.api.AbstractQubesAPI): self.fire_event_for_permission(pool=pool) other_info = '' + # Deprecated: remove this when all tools using this call are updated pool_size = pool.size if pool_size is not None: other_info += 'size={}\n'.format(pool_size) @@ -630,11 +631,6 @@ class QubesAdminAPI(qubes.api.AbstractQubesAPI): if pool_usage is not None: other_info += 'usage={}\n'.format(pool_usage) - pool_details = pool.usage_details - for name in pool_details: - if name not in ['data_size', 'data_usage']: - other_info += '{}={}\n'.format(name, pool_details[name]) - try: included_in = pool.included_in(self.app) if included_in: @@ -646,6 +642,26 @@ class QubesAdminAPI(qubes.api.AbstractQubesAPI): for prop, val in sorted(pool.config.items())) + \ other_info + @qubes.api.method('admin.pool.UsageDetails', no_payload=True, + scope='global', read=True) + @asyncio.coroutine + def pool_usage(self): + self.enforce(self.dest.name == 'dom0') + self.enforce(self.arg in self.app.pools.keys()) + + pool = self.app.pools[self.arg] + + self.fire_event_for_permission(pool=pool) + + usage = '' + + pool_details = pool.usage_details + + for name in sorted(pool_details): + usage += '{}={}\n'.format(name, pool_details[name]) + + return usage + @qubes.api.method('admin.pool.Add', scope='global', write=True) @asyncio.coroutine diff --git a/qubes/storage/lvm.py b/qubes/storage/lvm.py index 1c502d5e..b0a3b52d 100644 --- a/qubes/storage/lvm.py +++ b/qubes/storage/lvm.py @@ -207,8 +207,8 @@ class ThinPool(qubes.storage.Pool): metadata_usage = qubes.storage.lvm.size_cache[ self.volume_group + '/' + self.thin_pool]['metadata_usage'] except KeyError: - metadata_size = None - metadata_usage = None + metadata_size = 0 + metadata_usage = 0 result['metadata_size'] = metadata_size result['metadata_usage'] = metadata_usage diff --git a/qubes/tests/api_admin.py b/qubes/tests/api_admin.py index 5db6225b..47153ce6 100644 --- a/qubes/tests/api_admin.py +++ b/qubes/tests/api_admin.py @@ -582,14 +582,13 @@ class TC_00_VMs(AdminAPITestCase): 'pool1': unittest.mock.Mock(config={ 'param1': 'value1', 'param2': 'value2'}, usage=102400, - size=204800, - usage_details={'metadata_size': 500}) + size=204800) } self.app.pools['pool1'].included_in.return_value = None value = self.call_mgmt_func(b'admin.pool.Info', b'dom0', b'pool1') self.assertEqual(value, - 'param1=value1\nparam2=value2\nsize=204800\nusage=102400\nmetadata_size=500\n') + 'param1=value1\nparam2=value2\nsize=204800\nusage=102400\n') self.assertFalse(self.app.save.called) def test_151_pool_info_unsupported_size(self): @@ -623,6 +622,23 @@ class TC_00_VMs(AdminAPITestCase): '\nincluded_in=pool1\n') self.assertFalse(self.app.save.called) + def test_153_pool_usage(self): + self.app.pools = { + 'pool1': unittest.mock.Mock(config={ + 'param1': 'value1', 'param2': 'value2'}, + usage_details={ + 'data_usage': 102400, + 'data_size': 204800, + 'metadata_size': 1024, + 'metadata_usage': 50}) + } + self.app.pools['pool1'].included_in.return_value = None + value = self.call_mgmt_func(b'admin.pool.UsageDetails', b'dom0', b'pool1') + + self.assertEqual(value, + 'data_size=204800\ndata_usage=102400\nmetadata_size=1024\nmetadata_usage=50\n') + self.assertFalse(self.app.save.called) + @unittest.mock.patch('qubes.storage.pool_drivers') @unittest.mock.patch('qubes.storage.driver_parameters') def test_160_pool_add(self, mock_parameters, mock_drivers): diff --git a/qubes/tests/storage_lvm.py b/qubes/tests/storage_lvm.py index 5837395c..563fdf9f 100644 --- a/qubes/tests/storage_lvm.py +++ b/qubes/tests/storage_lvm.py @@ -997,6 +997,38 @@ class TC_00_ThinPool(ThinPoolBase): self.assertNotIn(volume1, list(self.pool.volumes)) self.assertNotIn(volume1, list(self.pool.volumes)) + def test_110_metadata_size(self): + with self.assertNotRaises(NotImplementedError): + usage = self.pool.usage_details + + metadata_size = usage['metadata_size'] + environ = os.environ.copy() + environ['LC_ALL'] = 'C.utf8' + pool_size = subprocess.check_output(['sudo', 'lvs', '--noheadings', + '-o', 'lv_metadata_size', + '--units', 'b', + self.pool.volume_group + '/' + self.pool.thin_pool], + env=environ) + self.assertEqual(metadata_size, int(pool_size.strip()[:-1])) + + def test_111_metadata_usage(self): + with self.assertNotRaises(NotImplementedError): + usage = self.pool.usage_details + + metadata_usage = usage['metadata_usage'] + environ = os.environ.copy() + environ['LC_ALL'] = 'C.utf8' + + pool_info = subprocess.check_output(['sudo', 'lvs', '--noheadings', + '-o', 'lv_metadata_size,metadata_percent', + '--units', 'b', self.pool.volume_group + '/' + self.pool.thin_pool], + env=environ) + pool_size, pool_usage = pool_info.strip().split() + pool_size = int(pool_size[:-1]) + pool_usage = float(pool_usage) + self.assertEqual(metadata_usage, int(pool_size * pool_usage / 100)) + + @skipUnlessLvmPoolExists class TC_01_ThinPool(ThinPoolBase, qubes.tests.SystemTestCase): ''' Sanity tests for :py:class:`qubes.storage.lvm.ThinPool` '''