
Change-Id: I1b60add4dadf2eecfe582fceb18acb614b67f7f8 Signed-off-by: Julien Danjou <julien@danjou.info>
309 lines
13 KiB
Python
309 lines
13 KiB
Python
# -*- encoding: utf-8 -*-
|
|
#
|
|
# Copyright © 2012 eNovance <licensing@enovance.com>
|
|
#
|
|
# Author: Julien Danjou <julien@danjou.info>
|
|
#
|
|
# 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.
|
|
|
|
import copy
|
|
import datetime
|
|
|
|
from lxml import etree
|
|
|
|
try:
|
|
from nova import config as nova_config
|
|
except ImportError:
|
|
# NOTE(dhellmann): We want to try to maintain compatibility
|
|
# with folsom for the time being, so set the name nova_config
|
|
# to a sentinal we can use to trigger different behavior
|
|
# when we try to set up the configuration object.
|
|
from nova import flags
|
|
nova_config = False
|
|
# Import this to register compute_driver flag in Folsom
|
|
import nova.compute.manager
|
|
from nova.virt import driver
|
|
from ceilometer import counter
|
|
from ceilometer.compute import plugin
|
|
from ceilometer.compute import instance as compute_instance
|
|
from ceilometer.openstack.common import importutils
|
|
from ceilometer.openstack.common import log
|
|
from ceilometer.openstack.common import timeutils
|
|
|
|
|
|
def _instance_name(instance):
|
|
"""Shortcut to get instance name"""
|
|
return getattr(instance, 'OS-EXT-SRV-ATTR:instance_name', None)
|
|
|
|
|
|
def get_compute_driver():
|
|
# FIXME(jd) This function is made to be destroyed by an abstraction
|
|
# layer in Nova providing an hypervisor agnostic API.
|
|
# XXX(jd) Folsom compat
|
|
if not nova_config:
|
|
flags.parse_args([])
|
|
return flags.FLAGS.compute_driver
|
|
nova_config.parse_args([])
|
|
return nova_config.cfg.CONF.compute_driver or ""
|
|
|
|
|
|
def get_libvirt_connection():
|
|
"""Return an open connection for talking to libvirt."""
|
|
# The direct-import implementation only works with Folsom because
|
|
# the configuration setting changed.
|
|
try:
|
|
try:
|
|
return driver.load_compute_driver(None)
|
|
except AttributeError:
|
|
return importutils.import_object_ns('nova.virt',
|
|
get_compute_driver())
|
|
except ImportError:
|
|
# Fall back to the way it was done in Essex.
|
|
import nova.virt.connection
|
|
return nova.virt.connection.get_connection(read_only=True)
|
|
|
|
|
|
def make_counter_from_instance(instance, name, type, volume):
|
|
return counter.Counter(
|
|
name=name,
|
|
type=type,
|
|
volume=volume,
|
|
user_id=instance.user_id,
|
|
project_id=instance.tenant_id,
|
|
resource_id=instance.id,
|
|
timestamp=timeutils.isotime(),
|
|
resource_metadata=compute_instance.get_metadata_from_object(instance),
|
|
)
|
|
|
|
|
|
class LibVirtPollster(plugin.ComputePollster):
|
|
|
|
def is_enabled(self):
|
|
# XXX(llu): Keeps Folsom compatibility
|
|
try:
|
|
return driver.compute_driver_matches("LibvirtDriver")
|
|
except AttributeError:
|
|
# Use a fairly liberal substring check.
|
|
return 'libvirt' in get_compute_driver().lower()
|
|
|
|
|
|
class InstancePollster(LibVirtPollster):
|
|
|
|
def get_counters(self, manager, instance):
|
|
yield make_counter_from_instance(instance,
|
|
name='instance',
|
|
type=counter.TYPE_GAUGE,
|
|
volume=1,
|
|
)
|
|
yield make_counter_from_instance(instance,
|
|
name='instance:%s' %
|
|
instance.flavor['name'],
|
|
type=counter.TYPE_GAUGE,
|
|
volume=1,
|
|
)
|
|
|
|
|
|
class DiskIOPollster(LibVirtPollster):
|
|
|
|
LOG = log.getLogger(__name__ + '.diskio')
|
|
|
|
DISKIO_USAGE_MESSAGE = ' '.join(["DISKIO USAGE:",
|
|
"%s %s:",
|
|
"read-requests=%d",
|
|
"read-bytes=%d",
|
|
"write-requests=%d",
|
|
"write-bytes=%d",
|
|
"errors=%d",
|
|
])
|
|
|
|
def get_counters(self, manager, instance):
|
|
conn = get_libvirt_connection()
|
|
instance_name = _instance_name(instance)
|
|
try:
|
|
disks = conn.get_disks(instance_name)
|
|
except Exception as err:
|
|
self.LOG.warning('Ignoring instance %s: %s',
|
|
instance_name, err)
|
|
self.LOG.exception(err)
|
|
else:
|
|
r_bytes = 0
|
|
r_requests = 0
|
|
w_bytes = 0
|
|
w_requests = 0
|
|
for disk in disks:
|
|
stats = conn.block_stats(instance_name, disk)
|
|
self.LOG.info(self.DISKIO_USAGE_MESSAGE,
|
|
instance, disk, stats[0], stats[1],
|
|
stats[2], stats[3], stats[4])
|
|
r_bytes += stats[0]
|
|
r_requests += stats[1]
|
|
w_bytes += stats[3]
|
|
w_requests += stats[2]
|
|
yield make_counter_from_instance(instance,
|
|
name='disk.read.requests',
|
|
type=counter.TYPE_CUMULATIVE,
|
|
volume=r_requests,
|
|
)
|
|
yield make_counter_from_instance(instance,
|
|
name='disk.read.bytes',
|
|
type=counter.TYPE_CUMULATIVE,
|
|
volume=r_bytes,
|
|
)
|
|
yield make_counter_from_instance(instance,
|
|
name='disk.write.requests',
|
|
type=counter.TYPE_CUMULATIVE,
|
|
volume=w_requests,
|
|
)
|
|
yield make_counter_from_instance(instance,
|
|
name='disk.write.bytes',
|
|
type=counter.TYPE_CUMULATIVE,
|
|
volume=w_bytes,
|
|
)
|
|
|
|
|
|
class CPUPollster(LibVirtPollster):
|
|
|
|
LOG = log.getLogger(__name__ + '.cpu')
|
|
|
|
utilization_map = {}
|
|
|
|
def get_cpu_util(self, instance, cpu_info):
|
|
prev_times = self.utilization_map.get(instance.id)
|
|
self.utilization_map[instance.id] = (cpu_info['cpu_time'],
|
|
datetime.datetime.now())
|
|
cpu_util = 0.0
|
|
if prev_times:
|
|
prev_cpu = prev_times[0]
|
|
prev_timestamp = prev_times[1]
|
|
delta = self.utilization_map[instance.id][1] - prev_timestamp
|
|
elapsed = (delta.seconds * (10 ** 6) + delta.microseconds) * 1000
|
|
cores_fraction = 1.0 / cpu_info['num_cpu']
|
|
# account for cpu_time being reset when the instance is restarted
|
|
time_used = (cpu_info['cpu_time'] - prev_cpu
|
|
if prev_cpu <= cpu_info['cpu_time'] else
|
|
cpu_info['cpu_time'])
|
|
cpu_util = 100 * cores_fraction * time_used / elapsed
|
|
return cpu_util
|
|
|
|
def get_counters(self, manager, instance):
|
|
conn = get_libvirt_connection()
|
|
self.LOG.info('checking instance %s', instance.id)
|
|
try:
|
|
cpu_info = conn.get_info({'name': _instance_name(instance)})
|
|
self.LOG.info("CPUTIME USAGE: %s %d",
|
|
instance.__dict__, cpu_info['cpu_time'])
|
|
cpu_util = self.get_cpu_util(instance, cpu_info)
|
|
self.LOG.info("CPU UTILIZATION %%: %s %0.2f",
|
|
instance.__dict__, cpu_util)
|
|
# FIXME(eglynn): once we have a way of configuring which measures
|
|
# are published to each sink, we should by default
|
|
# disable publishing this derived measure to the
|
|
# metering store, only publishing to those sinks
|
|
# that specifically need it
|
|
yield make_counter_from_instance(instance,
|
|
name='cpu_util',
|
|
type=counter.TYPE_GAUGE,
|
|
volume=cpu_util,
|
|
)
|
|
yield make_counter_from_instance(instance,
|
|
name='cpu',
|
|
type=counter.TYPE_CUMULATIVE,
|
|
volume=cpu_info['cpu_time'],
|
|
)
|
|
except Exception as err:
|
|
self.LOG.error('could not get CPU time for %s: %s',
|
|
instance.id, err)
|
|
self.LOG.exception(err)
|
|
|
|
|
|
class NetPollster(LibVirtPollster):
|
|
|
|
LOG = log.getLogger(__name__ + '.net')
|
|
|
|
NET_USAGE_MESSAGE = ' '.join(["NETWORK USAGE:", "%s %s:", "read-bytes=%d",
|
|
"write-bytes=%d"])
|
|
|
|
def _get_vnics(self, conn, instance):
|
|
"""Get disks of an instance, only used to bypass bug#998089."""
|
|
domain = conn._conn.lookupByName(_instance_name(instance))
|
|
tree = etree.fromstring(domain.XMLDesc(0))
|
|
vnics = []
|
|
for interface in tree.findall('devices/interface'):
|
|
vnic = {}
|
|
vnic['name'] = interface.find('target').get('dev')
|
|
vnic['mac'] = interface.find('mac').get('address')
|
|
vnic['fref'] = interface.find('filterref').get('filter')
|
|
for param in interface.findall('filterref/parameter'):
|
|
vnic[param.get('name').lower()] = param.get('value')
|
|
vnics.append(vnic)
|
|
return vnics
|
|
|
|
@staticmethod
|
|
def make_vnic_counter(instance, name, type, volume, vnic_data):
|
|
resource_metadata = copy.copy(vnic_data)
|
|
resource_metadata['instance_id'] = instance.id
|
|
|
|
return counter.Counter(
|
|
name=name,
|
|
type=type,
|
|
volume=volume,
|
|
user_id=instance.user_id,
|
|
project_id=instance.tenant_id,
|
|
resource_id=vnic_data['fref'],
|
|
timestamp=timeutils.isotime(),
|
|
resource_metadata=resource_metadata
|
|
)
|
|
|
|
def get_counters(self, manager, instance):
|
|
conn = get_libvirt_connection()
|
|
instance_name = _instance_name(instance)
|
|
self.LOG.info('checking instance %s', instance.id)
|
|
try:
|
|
vnics = self._get_vnics(conn, instance)
|
|
except Exception as err:
|
|
self.LOG.warning('Ignoring instance %s: %s',
|
|
instance_name, err)
|
|
self.LOG.exception(err)
|
|
else:
|
|
domain = conn._conn.lookupByName(instance_name)
|
|
for vnic in vnics:
|
|
rx_bytes, rx_packets, _, _, \
|
|
tx_bytes, tx_packets, _, _ = \
|
|
domain.interfaceStats(vnic['name'])
|
|
self.LOG.info(self.NET_USAGE_MESSAGE, instance_name,
|
|
vnic['name'], rx_bytes, tx_bytes)
|
|
yield self.make_vnic_counter(instance,
|
|
name='network.incoming.bytes',
|
|
type=counter.TYPE_CUMULATIVE,
|
|
volume=rx_bytes,
|
|
vnic_data=vnic,
|
|
)
|
|
yield self.make_vnic_counter(instance,
|
|
name='network.outgoing.bytes',
|
|
type=counter.TYPE_CUMULATIVE,
|
|
volume=tx_bytes,
|
|
vnic_data=vnic,
|
|
)
|
|
yield self.make_vnic_counter(instance,
|
|
name='network.incoming.packets',
|
|
type=counter.TYPE_CUMULATIVE,
|
|
volume=rx_packets,
|
|
vnic_data=vnic,
|
|
)
|
|
yield self.make_vnic_counter(instance,
|
|
name='network.outgoing.packets',
|
|
type=counter.TYPE_CUMULATIVE,
|
|
volume=tx_packets,
|
|
vnic_data=vnic,
|
|
)
|