Allow to pass dict from resource discovery
Allowing to pass dict from resource discovery and storing it as metadata. Before it was only possible to pass IP address, which is not enough. For TripleO case, important metadata like image name and flavor from Nova are passed and stored in resource, allowing to make aggregate queries according to them. Change-Id: I4dabf79f248842398e1126d310d110b1fe95fad5 Closes-Bug: #1366871
This commit is contained in:
parent
926fc8c605
commit
9f0cfce9a6
@ -41,22 +41,36 @@ class NodesDiscoveryTripleO(plugin.DiscoveryBase):
|
|||||||
super(NodesDiscoveryTripleO, self).__init__()
|
super(NodesDiscoveryTripleO, self).__init__()
|
||||||
self.nova_cli = nova_client.Client()
|
self.nova_cli = nova_client.Client()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _address(instance, field):
|
||||||
|
return instance.addresses['ctlplane'][0].get(field)
|
||||||
|
|
||||||
def discover(self, param=None):
|
def discover(self, param=None):
|
||||||
"""Discover resources to monitor."""
|
"""Discover resources to monitor."""
|
||||||
|
|
||||||
instances = self.nova_cli.instance_get_all()
|
instances = self.nova_cli.instance_get_all()
|
||||||
ip_addresses = []
|
resources = []
|
||||||
for instance in instances:
|
for instance in instances:
|
||||||
try:
|
try:
|
||||||
ip_address = instance.addresses['ctlplane'][0]['addr']
|
ip_address = self._address(instance, 'addr')
|
||||||
final_address = (
|
final_address = (
|
||||||
cfg.CONF.hardware.url_scheme +
|
cfg.CONF.hardware.url_scheme +
|
||||||
cfg.CONF.hardware.readonly_user_name + ':' +
|
cfg.CONF.hardware.readonly_user_name + ':' +
|
||||||
cfg.CONF.hardware.readonly_user_password + '@' +
|
cfg.CONF.hardware.readonly_user_password + '@' +
|
||||||
ip_address)
|
ip_address)
|
||||||
ip_addresses.append(final_address)
|
|
||||||
|
resource = {
|
||||||
|
'resource_id': instance.id,
|
||||||
|
'resource_url': final_address,
|
||||||
|
'mac_addr': self._address(instance,
|
||||||
|
'OS-EXT-IPS-MAC:mac_addr'),
|
||||||
|
'image_id': instance.image['id'],
|
||||||
|
'flavor_id': instance.flavor['id']
|
||||||
|
}
|
||||||
|
|
||||||
|
resources.append(resource)
|
||||||
except KeyError:
|
except KeyError:
|
||||||
LOG.error(_("Couldn't obtain IP address of"
|
LOG.error(_("Couldn't obtain IP address of"
|
||||||
"instance %s") % instance.id)
|
"instance %s") % instance.id)
|
||||||
|
|
||||||
return ip_addresses
|
return resources
|
||||||
|
@ -25,14 +25,15 @@ import six
|
|||||||
@six.add_metaclass(abc.ABCMeta)
|
@six.add_metaclass(abc.ABCMeta)
|
||||||
class Inspector(object):
|
class Inspector(object):
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def inspect_generic(self, host, identifier, cache):
|
def inspect_generic(self, host, identifier, cache, extra_metadata=None):
|
||||||
"""A generic inspect function.
|
"""A generic inspect function.
|
||||||
|
|
||||||
:param host: the target host
|
:param host: the target host
|
||||||
:param identifier: the identifier of the metric
|
:param identifier: the identifier of the metric
|
||||||
:param cache: cache passed from the pollster
|
:param cache: cache passed from the pollster
|
||||||
|
:param extra_metadata: extra dict to be used as metadata
|
||||||
:return: an iterator of (value, metadata, extra)
|
:return: an iterator of (value, metadata, extra)
|
||||||
:return value: the sample value
|
:return value: the sample value
|
||||||
:return metadata: dict to construct sample's metadata
|
:return metadata: dict to construct sample's metadata
|
||||||
:return extra: dict of extra info to help constructing sample
|
:return extra: dict of extra metadata to help constructing sample
|
||||||
"""
|
"""
|
||||||
|
@ -354,7 +354,7 @@ class SNMPInspector(base.Inspector):
|
|||||||
new_oids.append(metadata[0])
|
new_oids.append(metadata[0])
|
||||||
return new_oids
|
return new_oids
|
||||||
|
|
||||||
def inspect_generic(self, host, identifier, cache):
|
def inspect_generic(self, host, identifier, cache, extra_metadata=None):
|
||||||
# the snmp definition for the corresponding meter
|
# the snmp definition for the corresponding meter
|
||||||
meter_def = self.MAPPING[identifier]
|
meter_def = self.MAPPING[identifier]
|
||||||
# collect oids that needs to be queried
|
# collect oids that needs to be queried
|
||||||
@ -373,6 +373,7 @@ class SNMPInspector(base.Inspector):
|
|||||||
meter_def['metric_oid'][0],
|
meter_def['metric_oid'][0],
|
||||||
meter_def['matching_type'],
|
meter_def['matching_type'],
|
||||||
False)
|
False)
|
||||||
|
extra_metadata = extra_metadata or {}
|
||||||
for oid in oids_for_sample_values:
|
for oid in oids_for_sample_values:
|
||||||
suffix = oid[len(meter_def['metric_oid'][0]):]
|
suffix = oid[len(meter_def['metric_oid'][0]):]
|
||||||
value = self.get_oid_value(oid_cache,
|
value = self.get_oid_value(oid_cache,
|
||||||
@ -382,15 +383,14 @@ class SNMPInspector(base.Inspector):
|
|||||||
metadata = self.construct_metadata(oid_cache,
|
metadata = self.construct_metadata(oid_cache,
|
||||||
meter_def['metadata'],
|
meter_def['metadata'],
|
||||||
suffix)
|
suffix)
|
||||||
extra = {}
|
|
||||||
# call post_op for special cases
|
# call post_op for special cases
|
||||||
if meter_def['post_op']:
|
if meter_def['post_op']:
|
||||||
func = getattr(self, meter_def['post_op'], None)
|
func = getattr(self, meter_def['post_op'], None)
|
||||||
if func:
|
if func:
|
||||||
value = func(host, cache, meter_def,
|
value = func(host, cache, meter_def,
|
||||||
value, metadata, extra,
|
value, metadata, extra_metadata,
|
||||||
suffix)
|
suffix)
|
||||||
yield (value, metadata, extra)
|
yield (value, metadata, extra_metadata)
|
||||||
|
|
||||||
def _post_op_net(self, host, cache, meter_def,
|
def _post_op_net(self, host, cache, meter_def,
|
||||||
value, metadata, extra, suffix):
|
value, metadata, extra, suffix):
|
||||||
|
@ -47,7 +47,36 @@ class HardwarePollster(plugin.CentralPollster):
|
|||||||
def default_discovery(self):
|
def default_discovery(self):
|
||||||
return 'tripleo_overcloud_nodes'
|
return 'tripleo_overcloud_nodes'
|
||||||
|
|
||||||
def get_samples(self, manager, cache, resources):
|
@staticmethod
|
||||||
|
def _parse_resource(res):
|
||||||
|
"""Parse resource from discovery.
|
||||||
|
|
||||||
|
Either URL can be given or dict. Dict has to contain at least
|
||||||
|
keys 'resource_id' and 'resource_url', all the dict keys will be stored
|
||||||
|
as metadata.
|
||||||
|
|
||||||
|
:param res: URL or dict containing all resource info.
|
||||||
|
:return parsed_url, resource_id, metadata: Returns parsed URL used for
|
||||||
|
SNMP query, unique identifier of the resource and metadata
|
||||||
|
of the resource.
|
||||||
|
"""
|
||||||
|
|
||||||
|
if isinstance(res, dict):
|
||||||
|
if 'resource_url' not in res or 'resource_id' not in res:
|
||||||
|
LOG.exception(_('Passed resource dict must contain keys '
|
||||||
|
'resource_id and resource_url.'))
|
||||||
|
|
||||||
|
metadata = res
|
||||||
|
parsed_url = netutils.urlsplit(res['resource_url'])
|
||||||
|
resource_id = res['resource_id']
|
||||||
|
else:
|
||||||
|
metadata = {}
|
||||||
|
parsed_url = netutils.urlsplit(res)
|
||||||
|
resource_id = res
|
||||||
|
|
||||||
|
return parsed_url, resource_id, metadata
|
||||||
|
|
||||||
|
def get_samples(self, manager, cache, resources=None):
|
||||||
"""Return an iterable of Sample instances from polling the resources.
|
"""Return an iterable of Sample instances from polling the resources.
|
||||||
|
|
||||||
:param manager: The service manager invoking the plugin
|
:param manager: The service manager invoking the plugin
|
||||||
@ -57,17 +86,19 @@ class HardwarePollster(plugin.CentralPollster):
|
|||||||
resources = resources or []
|
resources = resources or []
|
||||||
h_cache = cache.setdefault(self.CACHE_KEY, {})
|
h_cache = cache.setdefault(self.CACHE_KEY, {})
|
||||||
sample_iters = []
|
sample_iters = []
|
||||||
for res in resources:
|
for resource in resources:
|
||||||
parsed_url = netutils.urlsplit(res)
|
parsed_url, res, extra_metadata = self._parse_resource(resource)
|
||||||
ins = self._get_inspector(parsed_url)
|
ins = self._get_inspector(parsed_url)
|
||||||
try:
|
try:
|
||||||
# Call hardware inspector to poll for the data
|
# Call hardware inspector to poll for the data
|
||||||
i_cache = h_cache.setdefault(res, {})
|
i_cache = h_cache.setdefault(res, {})
|
||||||
|
|
||||||
if self.IDENTIFIER not in i_cache:
|
if self.IDENTIFIER not in i_cache:
|
||||||
i_cache[self.IDENTIFIER] = list(ins.inspect_generic(
|
i_cache[self.IDENTIFIER] = list(ins.inspect_generic(
|
||||||
parsed_url,
|
parsed_url,
|
||||||
self.IDENTIFIER,
|
self.IDENTIFIER,
|
||||||
i_cache))
|
i_cache,
|
||||||
|
extra_metadata))
|
||||||
# Generate samples
|
# Generate samples
|
||||||
if i_cache[self.IDENTIFIER]:
|
if i_cache[self.IDENTIFIER]:
|
||||||
sample_iters.append(self.generate_samples(
|
sample_iters.append(self.generate_samples(
|
||||||
|
@ -29,7 +29,7 @@ class _Base(plugin.HardwarePollster):
|
|||||||
|
|
||||||
def generate_one_sample(self, host, c_data):
|
def generate_one_sample(self, host, c_data):
|
||||||
value, metadata, extra = c_data
|
value, metadata, extra = c_data
|
||||||
res_id = host.hostname
|
res_id = extra.get('resource_id') or host.hostname
|
||||||
if metadata.get('device'):
|
if metadata.get('device'):
|
||||||
res_id = res_id + ".%s" % metadata.get('device')
|
res_id = res_id + ".%s" % metadata.get('device')
|
||||||
return util.make_sample_from_host(host,
|
return util.make_sample_from_host(host,
|
||||||
|
@ -29,7 +29,7 @@ class _Base(plugin.HardwarePollster):
|
|||||||
|
|
||||||
def generate_one_sample(self, host, c_data):
|
def generate_one_sample(self, host, c_data):
|
||||||
value, metadata, extra = c_data
|
value, metadata, extra = c_data
|
||||||
res_id = host.hostname
|
res_id = extra.get('resource_id') or host.hostname
|
||||||
if metadata.get('name'):
|
if metadata.get('name'):
|
||||||
res_id = res_id + ".%s" % metadata.get('name')
|
res_id = res_id + ".%s" % metadata.get('name')
|
||||||
return util.make_sample_from_host(host,
|
return util.make_sample_from_host(host,
|
||||||
|
@ -44,15 +44,16 @@ def make_sample_from_host(host_url, name, sample_type, unit, volume,
|
|||||||
res_metadata=None, extra={}):
|
res_metadata=None, extra={}):
|
||||||
|
|
||||||
resource_metadata = make_resource_metadata(res_metadata, host_url)
|
resource_metadata = make_resource_metadata(res_metadata, host_url)
|
||||||
|
resource_metadata.update(extra)
|
||||||
|
|
||||||
res_id = extra.get('resource_id') or resource_id or host_url.hostname
|
res_id = resource_id or extra.get('resource_id') or host_url.hostname
|
||||||
return sample.Sample(
|
return sample.Sample(
|
||||||
name='hardware.' + name,
|
name='hardware.' + name,
|
||||||
type=sample_type,
|
type=sample_type,
|
||||||
unit=unit,
|
unit=unit,
|
||||||
volume=volume,
|
volume=volume,
|
||||||
user_id=extra.get('user_id') or user_id,
|
user_id=user_id or extra.get('user_id'),
|
||||||
project_id=extra.get('project_id') or project_id,
|
project_id=project_id or extra.get('project_id'),
|
||||||
resource_id=res_id,
|
resource_id=res_id,
|
||||||
timestamp=timeutils.isotime(),
|
timestamp=timeutils.isotime(),
|
||||||
resource_metadata=resource_metadata,
|
resource_metadata=resource_metadata,
|
||||||
|
@ -42,7 +42,7 @@ class FakeInspector(inspector_base.Inspector):
|
|||||||
'disk.size.used': (90, disk_metadata, {}),
|
'disk.size.used': (90, disk_metadata, {}),
|
||||||
}
|
}
|
||||||
|
|
||||||
def inspect_generic(self, host, identifier, cache):
|
def inspect_generic(self, host, identifier, cache, extra_metadata=None):
|
||||||
yield self.DATA[identifier]
|
yield self.DATA[identifier]
|
||||||
|
|
||||||
|
|
||||||
|
@ -53,4 +53,9 @@ class TestPollsterUtils(test_base.BaseTestCase):
|
|||||||
extra=extra)
|
extra=extra)
|
||||||
self.assertEqual(None, s.user_id)
|
self.assertEqual(None, s.user_id)
|
||||||
self.assertEqual('project', s.project_id)
|
self.assertEqual('project', s.project_id)
|
||||||
self.assertEqual('resource', s.resource_id)
|
self.assertEqual('resource', s.resource_id)
|
||||||
|
self.assertEqual({'resource_url': 'snmp://127.0.0.1:161',
|
||||||
|
'project_id': 'project',
|
||||||
|
'resource_id':
|
||||||
|
'resource'},
|
||||||
|
s.resource_metadata)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user