bind_rndc.chart.py 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. # -*- coding: utf-8 -*-
  2. # Description: bind rndc netdata python.d module
  3. # Author: ilyam8
  4. # SPDX-License-Identifier: GPL-3.0-or-later
  5. import os
  6. from collections import defaultdict
  7. from subprocess import Popen
  8. from bases.FrameworkServices.SimpleService import SimpleService
  9. from bases.collection import find_binary
  10. update_every = 30
  11. ORDER = [
  12. 'name_server_statistics',
  13. 'incoming_queries',
  14. 'outgoing_queries',
  15. 'named_stats_size',
  16. ]
  17. CHARTS = {
  18. 'name_server_statistics': {
  19. 'options': [None, 'Name Server Statistics', 'stats', 'name server statistics',
  20. 'bind_rndc.name_server_statistics', 'line'],
  21. 'lines': [
  22. ['nms_requests', 'requests', 'incremental'],
  23. ['nms_rejected_queries', 'rejected_queries', 'incremental'],
  24. ['nms_success', 'success', 'incremental'],
  25. ['nms_failure', 'failure', 'incremental'],
  26. ['nms_responses', 'responses', 'incremental'],
  27. ['nms_duplicate', 'duplicate', 'incremental'],
  28. ['nms_recursion', 'recursion', 'incremental'],
  29. ['nms_nxrrset', 'nxrrset', 'incremental'],
  30. ['nms_nxdomain', 'nxdomain', 'incremental'],
  31. ['nms_non_auth_answer', 'non_auth_answer', 'incremental'],
  32. ['nms_auth_answer', 'auth_answer', 'incremental'],
  33. ['nms_dropped_queries', 'dropped_queries', 'incremental'],
  34. ]},
  35. 'incoming_queries': {
  36. 'options': [None, 'Incoming Queries', 'queries', 'incoming queries', 'bind_rndc.incoming_queries', 'line'],
  37. 'lines': [
  38. ]},
  39. 'outgoing_queries': {
  40. 'options': [None, 'Outgoing Queries', 'queries', 'outgoing queries', 'bind_rndc.outgoing_queries', 'line'],
  41. 'lines': [
  42. ]},
  43. 'named_stats_size': {
  44. 'options': [None, 'Named Stats File Size', 'MiB', 'file size', 'bind_rndc.stats_size', 'line'],
  45. 'lines': [
  46. ['stats_size', None, 'absolute', 1, 1 << 20]
  47. ]
  48. }
  49. }
  50. NMS = {
  51. 'nms_requests': [
  52. 'IPv4 requests received',
  53. 'IPv6 requests received',
  54. 'TCP requests received',
  55. 'requests with EDNS(0) receive'
  56. ],
  57. 'nms_responses': [
  58. 'responses sent',
  59. 'truncated responses sent',
  60. 'responses with EDNS(0) sent',
  61. 'requests with unsupported EDNS version received'
  62. ],
  63. 'nms_failure': [
  64. 'other query failures',
  65. 'queries resulted in SERVFAIL'
  66. ],
  67. 'nms_auth_answer': ['queries resulted in authoritative answer'],
  68. 'nms_non_auth_answer': ['queries resulted in non authoritative answer'],
  69. 'nms_nxrrset': ['queries resulted in nxrrset'],
  70. 'nms_success': ['queries resulted in successful answer'],
  71. 'nms_nxdomain': ['queries resulted in NXDOMAIN'],
  72. 'nms_recursion': ['queries caused recursion'],
  73. 'nms_duplicate': ['duplicate queries received'],
  74. 'nms_rejected_queries': [
  75. 'auth queries rejected',
  76. 'recursive queries rejected'
  77. ],
  78. 'nms_dropped_queries': ['queries dropped']
  79. }
  80. STATS = ['Name Server Statistics', 'Incoming Queries', 'Outgoing Queries']
  81. class Service(SimpleService):
  82. def __init__(self, configuration=None, name=None):
  83. SimpleService.__init__(self, configuration=configuration, name=name)
  84. self.order = ORDER
  85. self.definitions = CHARTS
  86. self.named_stats_path = self.configuration.get('named_stats_path', '/var/log/bind/named.stats')
  87. self.rndc = find_binary('rndc')
  88. self.data = dict(
  89. nms_requests=0,
  90. nms_responses=0,
  91. nms_failure=0,
  92. nms_auth=0,
  93. nms_non_auth=0,
  94. nms_nxrrset=0,
  95. nms_success=0,
  96. nms_nxdomain=0,
  97. nms_recursion=0,
  98. nms_duplicate=0,
  99. nms_rejected_queries=0,
  100. nms_dropped_queries=0,
  101. )
  102. def check(self):
  103. if not self.rndc:
  104. self.error('Can\'t locate "rndc" binary or binary is not executable by netdata')
  105. return False
  106. if not (os.path.isfile(self.named_stats_path) and os.access(self.named_stats_path, os.R_OK)):
  107. self.error('Cannot access file %s' % self.named_stats_path)
  108. return False
  109. run_rndc = Popen([self.rndc, 'stats'], shell=False)
  110. run_rndc.wait()
  111. if not run_rndc.returncode:
  112. return True
  113. self.error('Not enough permissions to run "%s stats"' % self.rndc)
  114. return False
  115. def _get_raw_data(self):
  116. """
  117. Run 'rndc stats' and read last dump from named.stats
  118. :return: dict
  119. """
  120. result = dict()
  121. try:
  122. current_size = os.path.getsize(self.named_stats_path)
  123. run_rndc = Popen([self.rndc, 'stats'], shell=False)
  124. run_rndc.wait()
  125. if run_rndc.returncode:
  126. return None
  127. with open(self.named_stats_path) as named_stats:
  128. named_stats.seek(current_size)
  129. result['stats'] = named_stats.readlines()
  130. result['size'] = current_size
  131. return result
  132. except (OSError, IOError):
  133. return None
  134. def _get_data(self):
  135. """
  136. Parse data from _get_raw_data()
  137. :return: dict
  138. """
  139. raw_data = self._get_raw_data()
  140. if raw_data is None:
  141. return None
  142. parsed = dict()
  143. for stat in STATS:
  144. parsed[stat] = parse_stats(field=stat,
  145. named_stats=raw_data['stats'])
  146. self.data.update(nms_mapper(data=parsed['Name Server Statistics']))
  147. for elem in zip(['Incoming Queries', 'Outgoing Queries'], ['incoming_queries', 'outgoing_queries']):
  148. parsed_key, chart_name = elem[0], elem[1]
  149. for dimension_id, value in queries_mapper(data=parsed[parsed_key],
  150. add=chart_name[:9]).items():
  151. if dimension_id not in self.data:
  152. dimension = dimension_id.replace(chart_name[:9], '')
  153. if dimension_id not in self.charts[chart_name]:
  154. self.charts[chart_name].add_dimension([dimension_id, dimension, 'incremental'])
  155. self.data[dimension_id] = value
  156. self.data['stats_size'] = raw_data['size']
  157. return self.data
  158. def parse_stats(field, named_stats):
  159. """
  160. :param field: str:
  161. :param named_stats: list:
  162. :return: dict
  163. Example:
  164. filed: 'Incoming Queries'
  165. names_stats (list of lines):
  166. ++ Incoming Requests ++
  167. 1405660 QUERY
  168. 3 NOTIFY
  169. ++ Incoming Queries ++
  170. 1214961 A
  171. 75 NS
  172. 2 CNAME
  173. 2897 SOA
  174. 35544 PTR
  175. 14 MX
  176. 5822 TXT
  177. 145974 AAAA
  178. 371 SRV
  179. ++ Outgoing Queries ++
  180. ...
  181. result:
  182. {'A', 1214961, 'NS': 75, 'CNAME': 2, 'SOA': 2897, ...}
  183. """
  184. data = dict()
  185. ns = iter(named_stats)
  186. for line in ns:
  187. if field not in line:
  188. continue
  189. while True:
  190. try:
  191. line = next(ns)
  192. except StopIteration:
  193. break
  194. if '++' not in line:
  195. if '[' in line:
  196. continue
  197. v, k = line.strip().split(' ', 1)
  198. if k not in data:
  199. data[k] = 0
  200. data[k] += int(v)
  201. continue
  202. break
  203. break
  204. return data
  205. def nms_mapper(data):
  206. """
  207. :param data: dict
  208. :return: dict(defaultdict)
  209. """
  210. result = defaultdict(int)
  211. for k, v in NMS.items():
  212. for elem in v:
  213. result[k] += data.get(elem, 0)
  214. return result
  215. def queries_mapper(data, add):
  216. """
  217. :param data: dict
  218. :param add: str
  219. :return: dict
  220. """
  221. return dict([(add + k, v) for k, v in data.items()])