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):
|
||||
instance_name = util.instance_name(instance)
|
||||
vm_ref = self._lookup_by_name(instance_name)
|
||||
metrics_ref = self._call_xenapi("VM.get_metrics", vm_ref)
|
||||
metrics_rec = self._call_xenapi("VM_metrics.get_record",
|
||||
metrics_ref)
|
||||
# Stat provided from XenServer is in B, converting it to MB.
|
||||
memory = int(metrics_rec['memory_actual']) / units.Mi
|
||||
return virt_inspector.MemoryUsageStats(usage=memory)
|
||||
total_mem = float(self._call_xenapi("VM.query_data_source",
|
||||
vm_ref,
|
||||
"memory"))
|
||||
try:
|
||||
free_mem = float(self._call_xenapi("VM.query_data_source",
|
||||
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):
|
||||
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.xenapi import inspector as xenapi_inspector
|
||||
from ceilometer.tests.unit.compute.virt.xenapi import fake_XenAPI
|
||||
|
||||
|
||||
class TestSwapXapiHost(base.BaseTestCase):
|
||||
@ -81,25 +82,59 @@ class TestXenapiInspection(base.BaseTestCase):
|
||||
def test_inspect_memory_usage(self):
|
||||
fake_instance = {'OS-EXT-SRV-ATTR:instance_name': 'fake_instance_name',
|
||||
'id': 'fake_instance_id'}
|
||||
fake_stat = virt_inspector.MemoryUsageStats(usage=128)
|
||||
fake_stat = virt_inspector.MemoryUsageStats(usage=64)
|
||||
|
||||
def fake_xenapi_request(method, args):
|
||||
metrics_rec = {
|
||||
'memory_actual': '134217728',
|
||||
}
|
||||
def _fake_xenapi_request(method, args):
|
||||
fake_total_mem = 134217728.0
|
||||
fake_free_mem = 65536.0
|
||||
|
||||
if method == 'VM.get_by_name_label':
|
||||
return ['vm_ref']
|
||||
elif method == 'VM.get_metrics':
|
||||
return 'metrics_ref'
|
||||
elif method == 'VM_metrics.get_record':
|
||||
return metrics_rec
|
||||
elif method == 'VM.query_data_source':
|
||||
if 'memory' in args:
|
||||
return fake_total_mem
|
||||
elif 'memory_internal_free' in args:
|
||||
return fake_free_mem
|
||||
else:
|
||||
return None
|
||||
else:
|
||||
return None
|
||||
|
||||
session = self.inspector.session
|
||||
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)
|
||||
self.assertEqual(fake_stat, memory_stat)
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user