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:
parent
bf38c2c932
commit
d38c580e0a
@ -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)
|
||||||
|
19
ceilometer/tests/unit/compute/virt/xenapi/fake_XenAPI.py
Normal file
19
ceilometer/tests/unit/compute/virt/xenapi/fake_XenAPI.py
Normal 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
|
@ -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)
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user