netdata/collectors/python.d.plugin/monit/monit.chart.py

179 lines
6.4 KiB
Python

# -*- coding: utf-8 -*-
# Description: monit netdata python.d module
# Author: Evgeniy K. (n0guest)
# SPDX-License-Identifier: GPL-3.0-or-later
import xml.etree.ElementTree as ET
from bases.FrameworkServices.UrlService import UrlService
# see enum State_Type from monit.h (https://bitbucket.org/tildeslash/monit/src/master/src/monit.h)
MONIT_SERVICE_NAMES = [
'Filesystem',
'Directory',
'File',
'Process',
'Host',
'System',
'Fifo',
'Program',
'Net',
]
DEFAULT_SERVICES_IDS = [0, 1, 2, 3, 4, 6, 7, 8]
# charts order (can be overridden if you want less charts, or different order)
ORDER = [
'filesystem',
'directory',
'file',
'process',
'process_uptime',
'process_threads',
'process_children',
'host',
'host_latency',
'system',
'fifo',
'program',
'net'
]
CHARTS = {
'filesystem': {
'options': ['filesystems', 'Filesystems', 'filesystems', 'filesystem', 'monit.filesystems', 'line'],
'lines': []
},
'directory': {
'options': ['directories', 'Directories', 'directories', 'filesystem', 'monit.directories', 'line'],
'lines': []
},
'file': {
'options': ['files', 'Files', 'files', 'filesystem', 'monit.files', 'line'],
'lines': []
},
'fifo': {
'options': ['fifos', 'Pipes (fifo)', 'pipes', 'filesystem', 'monit.fifos', 'line'],
'lines': []
},
'program': {
'options': ['programs', 'Programs statuses', 'programs', 'applications', 'monit.programs', 'line'],
'lines': []
},
'process': {
'options': ['processes', 'Processes statuses', 'processes', 'applications', 'monit.services', 'line'],
'lines': []
},
'process_uptime': {
'options': ['processes uptime', 'Processes uptime', 'seconds', 'applications',
'monit.process_uptime', 'line', 'hidden'],
'lines': []
},
'process_threads': {
'options': ['processes threads', 'Processes threads', 'threads', 'applications',
'monit.process_threads', 'line'],
'lines': []
},
'process_children': {
'options': ['processes childrens', 'Child processes', 'childrens', 'applications',
'monit.process_childrens', 'line'],
'lines': []
},
'host': {
'options': ['hosts', 'Hosts', 'hosts', 'network', 'monit.hosts', 'line'],
'lines': []
},
'host_latency': {
'options': ['hosts latency', 'Hosts latency', 'milliseconds/s', 'network', 'monit.host_latency', 'line'],
'lines': []
},
'net': {
'options': ['interfaces', 'Network interfaces and addresses', 'interfaces', 'network',
'monit.networks', 'line'],
'lines': []
},
}
class Service(UrlService):
def __init__(self, configuration=None, name=None):
UrlService.__init__(self, configuration=configuration, name=name)
self.order = ORDER
self.definitions = CHARTS
base_url = self.configuration.get('url', 'http://localhost:2812')
self.url = '{0}/_status?format=xml&level=full'.format(base_url)
def parse(self, data):
try:
xml = ET.fromstring(data)
except ET.ParseError:
self.error("URL {0} didn't return a vaild XML page. Please check your settings.".format(self.url))
return None
return xml
def check(self):
self._manager = self._build_manager()
raw_data = self._get_raw_data()
if not raw_data:
return None
return bool(self.parse(raw_data))
def _get_data(self):
raw_data = self._get_raw_data()
if not raw_data:
return None
xml = self.parse(raw_data)
if not xml:
return None
data = {}
for service_id in DEFAULT_SERVICES_IDS:
service_category = MONIT_SERVICE_NAMES[service_id].lower()
if service_category == 'system':
self.debug("Skipping service from 'System' category, because it's useless in graphs")
continue
xpath_query = "./service[@type='{0}']".format(service_id)
self.debug('Searching for {0} as {1}'.format(service_category, xpath_query))
for service_node in xml.findall(xpath_query):
service_name = service_node.find('name').text
service_status = service_node.find('status').text
service_monitoring = service_node.find('monitor').text
self.debug('=> found {0} with type={1}, status={2}, monitoring={3}'.format(service_name,
service_id, service_status, service_monitoring))
dimension_key = service_category + '_' + service_name
if dimension_key not in self.charts[service_category]:
self.charts[service_category].add_dimension([dimension_key, service_name, 'absolute'])
data[dimension_key] = 1 if service_status == '0' and service_monitoring == '1' else 0
if service_category == 'process':
for subnode in ('uptime', 'threads', 'children'):
subnode_value = service_node.find(subnode)
if subnode_value is None:
continue
if subnode == 'uptime' and int(subnode_value.text) < 0:
self.debug('Skipping bugged metrics with negative uptime (monit before v5.16')
continue
dimension_key = 'process_{0}_{1}'.format(subnode, service_name)
if dimension_key not in self.charts['process_' + subnode]:
self.charts['process_' + subnode].add_dimension([dimension_key, service_name, 'absolute'])
data[dimension_key] = int(subnode_value.text)
if service_category == 'host':
subnode_value = service_node.find('./icmp/responsetime')
if subnode_value is None:
continue
dimension_key = 'host_latency_{0}'.format(service_name)
if dimension_key not in self.charts['host_latency']:
self.charts['host_latency'].add_dimension([dimension_key, service_name,
'absolute', 1000, 1000000])
data[dimension_key] = float(subnode_value.text) * 1000000
return data or None