XenAPI: correct polling on memory_usage

Currently the memory_usage polled is the allocated memory size;
but it should be "Volume of RAM used by the instance from the amount
of its allocated memory" per the meter definition.

Change-Id: I5b1499a98d4e9873e5ae98c3b2cd430ac3c733eb
This commit is contained in:
Jianghua Wang 2016-09-06 16:34:51 +08:00
parent bf38c2c932
commit d38c580e0a
3 changed files with 81 additions and 16 deletions

View File

@ -135,12 +135,23 @@ class XenapiInspector(virt_inspector.Inspector):
def inspect_memory_usage(self, instance, duration=None): def inspect_memory_usage(self, instance, duration=None):
instance_name = util.instance_name(instance) instance_name = util.instance_name(instance)
vm_ref = self._lookup_by_name(instance_name) vm_ref = self._lookup_by_name(instance_name)
metrics_ref = self._call_xenapi("VM.get_metrics", vm_ref) total_mem = float(self._call_xenapi("VM.query_data_source",
metrics_rec = self._call_xenapi("VM_metrics.get_record", vm_ref,
metrics_ref) "memory"))
# Stat provided from XenServer is in B, converting it to MB. try:
memory = int(metrics_rec['memory_actual']) / units.Mi free_mem = float(self._call_xenapi("VM.query_data_source",
return virt_inspector.MemoryUsageStats(usage=memory) vm_ref,
"memory_internal_free"))
except api.Failure:
# If PV tools is not installed in the guest instance, it's
# impossible to get free memory. So give it a default value
# as 0.
free_mem = 0
# memory provided from XenServer is in Bytes;
# memory_internal_free provided from XenServer is in KB,
# converting it to MB.
memory_usage = (total_mem - free_mem * units.Ki) / units.Mi
return virt_inspector.MemoryUsageStats(usage=memory_usage)
def inspect_vnic_rates(self, instance, duration=None): def inspect_vnic_rates(self, instance, duration=None):
instance_name = util.instance_name(instance) instance_name = util.instance_name(instance)

View File

@ -0,0 +1,19 @@
# Copyright 2016 Citrix
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
"""fake XenAPI for testing in case XenAPI doesn't exist in the test env"""
class Failure(Exception):
def __init__(self, details):
self.details = details

View File

@ -19,6 +19,7 @@ from oslotest import base
from ceilometer.compute.virt import inspector as virt_inspector from ceilometer.compute.virt import inspector as virt_inspector
from ceilometer.compute.virt.xenapi import inspector as xenapi_inspector from ceilometer.compute.virt.xenapi import inspector as xenapi_inspector
from ceilometer.tests.unit.compute.virt.xenapi import fake_XenAPI
class TestSwapXapiHost(base.BaseTestCase): class TestSwapXapiHost(base.BaseTestCase):
@ -81,25 +82,59 @@ class TestXenapiInspection(base.BaseTestCase):
def test_inspect_memory_usage(self): def test_inspect_memory_usage(self):
fake_instance = {'OS-EXT-SRV-ATTR:instance_name': 'fake_instance_name', fake_instance = {'OS-EXT-SRV-ATTR:instance_name': 'fake_instance_name',
'id': 'fake_instance_id'} 'id': 'fake_instance_id'}
fake_stat = virt_inspector.MemoryUsageStats(usage=128) fake_stat = virt_inspector.MemoryUsageStats(usage=64)
def fake_xenapi_request(method, args): def _fake_xenapi_request(method, args):
metrics_rec = { fake_total_mem = 134217728.0
'memory_actual': '134217728', fake_free_mem = 65536.0
}
if method == 'VM.get_by_name_label': if method == 'VM.get_by_name_label':
return ['vm_ref'] return ['vm_ref']
elif method == 'VM.get_metrics': elif method == 'VM.query_data_source':
return 'metrics_ref' if 'memory' in args:
elif method == 'VM_metrics.get_record': return fake_total_mem
return metrics_rec elif 'memory_internal_free' in args:
return fake_free_mem
else:
return None
else: else:
return None return None
session = self.inspector.session session = self.inspector.session
with mock.patch.object(session, 'xenapi_request', with mock.patch.object(session, 'xenapi_request',
side_effect=fake_xenapi_request): side_effect=_fake_xenapi_request):
memory_stat = self.inspector.inspect_memory_usage(fake_instance)
self.assertEqual(fake_stat, memory_stat)
def test_inspect_memory_usage_without_freeMem(self):
fake_instance = {'OS-EXT-SRV-ATTR:instance_name': 'fake_instance_name',
'id': 'fake_instance_id'}
fake_stat = virt_inspector.MemoryUsageStats(usage=128)
def _fake_xenapi_request(method, args):
if xenapi_inspector.api is None:
# the XenAPI may not exist in the test environment.
# In that case, we use the fake XenAPI for testing.
xenapi_inspector.api = fake_XenAPI
fake_total_mem = 134217728.0
fake_details = ['INTERNAL_ERROR',
'Rrd.Invalid_data_source("memory_internal_free")']
if method == 'VM.get_by_name_label':
return ['vm_ref']
elif method == 'VM.query_data_source':
if 'memory' in args:
return fake_total_mem
elif 'memory_internal_free' in args:
raise xenapi_inspector.api.Failure(fake_details)
else:
return None
else:
return None
session = self.inspector.session
with mock.patch.object(session, 'xenapi_request',
side_effect=_fake_xenapi_request):
memory_stat = self.inspector.inspect_memory_usage(fake_instance) memory_stat = self.inspector.inspect_memory_usage(fake_instance)
self.assertEqual(fake_stat, memory_stat) self.assertEqual(fake_stat, memory_stat)