ntpd.chart.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385
  1. # -*- coding: utf-8 -*-
  2. # Description: ntpd netdata python.d module
  3. # Author: Sven Mäder (rda0)
  4. # Author: Ilya Mashchenko (ilyam8)
  5. # SPDX-License-Identifier: GPL-3.0-or-later
  6. import re
  7. import struct
  8. from bases.FrameworkServices.SocketService import SocketService
  9. # NTP Control Message Protocol constants
  10. MODE = 6
  11. HEADER_FORMAT = '!BBHHHHH'
  12. HEADER_LEN = 12
  13. OPCODES = {
  14. 'readstat': 1,
  15. 'readvar': 2
  16. }
  17. # Maximal dimension precision
  18. PRECISION = 1000000
  19. # Static charts
  20. ORDER = [
  21. 'sys_offset',
  22. 'sys_jitter',
  23. 'sys_frequency',
  24. 'sys_wander',
  25. 'sys_rootdelay',
  26. 'sys_rootdisp',
  27. 'sys_stratum',
  28. 'sys_tc',
  29. 'sys_precision',
  30. 'peer_offset',
  31. 'peer_delay',
  32. 'peer_dispersion',
  33. 'peer_jitter',
  34. 'peer_xleave',
  35. 'peer_rootdelay',
  36. 'peer_rootdisp',
  37. 'peer_stratum',
  38. 'peer_hmode',
  39. 'peer_pmode',
  40. 'peer_hpoll',
  41. 'peer_ppoll',
  42. 'peer_precision'
  43. ]
  44. CHARTS = {
  45. 'sys_offset': {
  46. 'options': [None, 'Combined offset of server relative to this host', 'milliseconds',
  47. 'system', 'ntpd.sys_offset', 'area'],
  48. 'lines': [
  49. ['offset', 'offset', 'absolute', 1, PRECISION]
  50. ]
  51. },
  52. 'sys_jitter': {
  53. 'options': [None, 'Combined system jitter and clock jitter', 'milliseconds',
  54. 'system', 'ntpd.sys_jitter', 'line'],
  55. 'lines': [
  56. ['sys_jitter', 'system', 'absolute', 1, PRECISION],
  57. ['clk_jitter', 'clock', 'absolute', 1, PRECISION]
  58. ]
  59. },
  60. 'sys_frequency': {
  61. 'options': [None, 'Frequency offset relative to hardware clock', 'ppm', 'system', 'ntpd.sys_frequency', 'area'],
  62. 'lines': [
  63. ['frequency', 'frequency', 'absolute', 1, PRECISION]
  64. ]
  65. },
  66. 'sys_wander': {
  67. 'options': [None, 'Clock frequency wander', 'ppm', 'system', 'ntpd.sys_wander', 'area'],
  68. 'lines': [
  69. ['clk_wander', 'clock', 'absolute', 1, PRECISION]
  70. ]
  71. },
  72. 'sys_rootdelay': {
  73. 'options': [None, 'Total roundtrip delay to the primary reference clock', 'milliseconds', 'system',
  74. 'ntpd.sys_rootdelay', 'area'],
  75. 'lines': [
  76. ['rootdelay', 'delay', 'absolute', 1, PRECISION]
  77. ]
  78. },
  79. 'sys_rootdisp': {
  80. 'options': [None, 'Total root dispersion to the primary reference clock', 'milliseconds', 'system',
  81. 'ntpd.sys_rootdisp', 'area'],
  82. 'lines': [
  83. ['rootdisp', 'dispersion', 'absolute', 1, PRECISION]
  84. ]
  85. },
  86. 'sys_stratum': {
  87. 'options': [None, 'Stratum (1-15)', 'stratum', 'system', 'ntpd.sys_stratum', 'line'],
  88. 'lines': [
  89. ['stratum', 'stratum', 'absolute', 1, PRECISION]
  90. ]
  91. },
  92. 'sys_tc': {
  93. 'options': [None, 'Time constant and poll exponent (3-17)', 'log2 s', 'system', 'ntpd.sys_tc', 'line'],
  94. 'lines': [
  95. ['tc', 'current', 'absolute', 1, PRECISION],
  96. ['mintc', 'minimum', 'absolute', 1, PRECISION]
  97. ]
  98. },
  99. 'sys_precision': {
  100. 'options': [None, 'Precision', 'log2 s', 'system', 'ntpd.sys_precision', 'line'],
  101. 'lines': [
  102. ['precision', 'precision', 'absolute', 1, PRECISION]
  103. ]
  104. }
  105. }
  106. PEER_CHARTS = {
  107. 'peer_offset': {
  108. 'options': [None, 'Filter offset', 'milliseconds', 'peers', 'ntpd.peer_offset', 'line'],
  109. 'lines': []
  110. },
  111. 'peer_delay': {
  112. 'options': [None, 'Filter delay', 'milliseconds', 'peers', 'ntpd.peer_delay', 'line'],
  113. 'lines': []
  114. },
  115. 'peer_dispersion': {
  116. 'options': [None, 'Filter dispersion', 'milliseconds', 'peers', 'ntpd.peer_dispersion', 'line'],
  117. 'lines': []
  118. },
  119. 'peer_jitter': {
  120. 'options': [None, 'Filter jitter', 'milliseconds', 'peers', 'ntpd.peer_jitter', 'line'],
  121. 'lines': []
  122. },
  123. 'peer_xleave': {
  124. 'options': [None, 'Interleave delay', 'milliseconds', 'peers', 'ntpd.peer_xleave', 'line'],
  125. 'lines': []
  126. },
  127. 'peer_rootdelay': {
  128. 'options': [None, 'Total roundtrip delay to the primary reference clock', 'milliseconds', 'peers',
  129. 'ntpd.peer_rootdelay', 'line'],
  130. 'lines': []
  131. },
  132. 'peer_rootdisp': {
  133. 'options': [None, 'Total root dispersion to the primary reference clock', 'ms', 'peers',
  134. 'ntpd.peer_rootdisp', 'line'],
  135. 'lines': []
  136. },
  137. 'peer_stratum': {
  138. 'options': [None, 'Stratum (1-15)', 'stratum', 'peers', 'ntpd.peer_stratum', 'line'],
  139. 'lines': []
  140. },
  141. 'peer_hmode': {
  142. 'options': [None, 'Host mode (1-6)', 'hmode', 'peers', 'ntpd.peer_hmode', 'line'],
  143. 'lines': []
  144. },
  145. 'peer_pmode': {
  146. 'options': [None, 'Peer mode (1-5)', 'pmode', 'peers', 'ntpd.peer_pmode', 'line'],
  147. 'lines': []
  148. },
  149. 'peer_hpoll': {
  150. 'options': [None, 'Host poll exponent', 'log2 s', 'peers', 'ntpd.peer_hpoll', 'line'],
  151. 'lines': []
  152. },
  153. 'peer_ppoll': {
  154. 'options': [None, 'Peer poll exponent', 'log2 s', 'peers', 'ntpd.peer_ppoll', 'line'],
  155. 'lines': []
  156. },
  157. 'peer_precision': {
  158. 'options': [None, 'Precision', 'log2 s', 'peers', 'ntpd.peer_precision', 'line'],
  159. 'lines': []
  160. }
  161. }
  162. class Base:
  163. regex = re.compile(r'([a-z_]+)=((?:-)?[0-9]+(?:\.[0-9]+)?)')
  164. @staticmethod
  165. def get_header(associd=0, operation='readvar'):
  166. """
  167. Constructs the NTP Control Message header:
  168. 0 1 2 3
  169. 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
  170. +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  171. |LI | VN |Mode |R|E|M| OpCode | Sequence Number |
  172. +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  173. | Status | Association ID |
  174. +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  175. | Offset | Count |
  176. +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  177. """
  178. version = 2
  179. sequence = 1
  180. status = 0
  181. offset = 0
  182. count = 0
  183. header = struct.pack(HEADER_FORMAT, (version << 3 | MODE), OPCODES[operation],
  184. sequence, status, associd, offset, count)
  185. return header
  186. class System(Base):
  187. def __init__(self):
  188. self.request = self.get_header()
  189. def get_data(self, raw):
  190. """
  191. Extracts key=value pairs with float/integer from ntp response packet data.
  192. """
  193. data = dict()
  194. for key, value in self.regex.findall(raw):
  195. data[key] = float(value) * PRECISION
  196. return data
  197. class Peer(Base):
  198. def __init__(self, idx, name):
  199. self.id = idx
  200. self.real_name = name
  201. self.name = name.replace('.', '_')
  202. self.request = self.get_header(self.id)
  203. def get_data(self, raw):
  204. """
  205. Extracts key=value pairs with float/integer from ntp response packet data.
  206. """
  207. data = dict()
  208. for key, value in self.regex.findall(raw):
  209. dimension = '_'.join([self.name, key])
  210. data[dimension] = float(value) * PRECISION
  211. return data
  212. class Service(SocketService):
  213. def __init__(self, configuration=None, name=None):
  214. SocketService.__init__(self, configuration=configuration, name=name)
  215. self.order = list(ORDER)
  216. self.definitions = dict(CHARTS)
  217. self.port = 'ntp'
  218. self.dgram_socket = True
  219. self.system = System()
  220. self.peers = dict()
  221. self.request = str()
  222. self.retries = 0
  223. self.show_peers = self.configuration.get('show_peers', False)
  224. self.peer_rescan = self.configuration.get('peer_rescan', 60)
  225. if self.show_peers:
  226. self.definitions.update(PEER_CHARTS)
  227. def check(self):
  228. """
  229. Checks if we can get valid systemvars.
  230. If not, returns None to disable module.
  231. """
  232. self._parse_config()
  233. peer_filter = self.configuration.get('peer_filter', r'127\..*')
  234. try:
  235. self.peer_filter = re.compile(r'^((0\.0\.0\.0)|({0}))$'.format(peer_filter))
  236. except re.error as error:
  237. self.error('Compile pattern error (peer_filter) : {0}'.format(error))
  238. return None
  239. self.request = self.system.request
  240. raw_systemvars = self._get_raw_data()
  241. if not self.system.get_data(raw_systemvars):
  242. return None
  243. return True
  244. def get_data(self):
  245. """
  246. Gets systemvars data on each update.
  247. Gets peervars data for all peers on each update.
  248. """
  249. data = dict()
  250. self.request = self.system.request
  251. raw = self._get_raw_data()
  252. if not raw:
  253. return None
  254. data.update(self.system.get_data(raw))
  255. if not self.show_peers:
  256. return data
  257. if not self.peers or self.runs_counter % self.peer_rescan == 0 or self.retries > 8:
  258. self.find_new_peers()
  259. for peer in self.peers.values():
  260. self.request = peer.request
  261. peer_data = peer.get_data(self._get_raw_data())
  262. if peer_data:
  263. data.update(peer_data)
  264. else:
  265. self.retries += 1
  266. return data
  267. def find_new_peers(self):
  268. new_peers = dict((p.real_name, p) for p in self.get_peers())
  269. if new_peers:
  270. peers_to_remove = set(self.peers) - set(new_peers)
  271. peers_to_add = set(new_peers) - set(self.peers)
  272. for peer_name in peers_to_remove:
  273. self.hide_old_peer_from_charts(self.peers[peer_name])
  274. del self.peers[peer_name]
  275. for peer_name in peers_to_add:
  276. self.add_new_peer_to_charts(new_peers[peer_name])
  277. self.peers.update(new_peers)
  278. self.retries = 0
  279. def add_new_peer_to_charts(self, peer):
  280. for chart_id in set(self.charts.charts) & set(PEER_CHARTS):
  281. dim_id = peer.name + chart_id[4:]
  282. if dim_id not in self.charts[chart_id]:
  283. self.charts[chart_id].add_dimension([dim_id, peer.real_name, 'absolute', 1, PRECISION])
  284. else:
  285. self.charts[chart_id].hide_dimension(dim_id, reverse=True)
  286. def hide_old_peer_from_charts(self, peer):
  287. for chart_id in set(self.charts.charts) & set(PEER_CHARTS):
  288. dim_id = peer.name + chart_id[4:]
  289. self.charts[chart_id].hide_dimension(dim_id)
  290. def get_peers(self):
  291. self.request = Base.get_header(operation='readstat')
  292. raw_data = self._get_raw_data(raw=True)
  293. if not raw_data:
  294. return list()
  295. peer_ids = self.get_peer_ids(raw_data)
  296. if not peer_ids:
  297. return list()
  298. new_peers = list()
  299. for peer_id in peer_ids:
  300. self.request = Base.get_header(peer_id)
  301. raw_peer_data = self._get_raw_data()
  302. if not raw_peer_data:
  303. continue
  304. srcadr = re.search(r'(srcadr)=([^,]+)', raw_peer_data)
  305. if not srcadr:
  306. continue
  307. srcadr = srcadr.group(2)
  308. if self.peer_filter.search(srcadr):
  309. continue
  310. stratum = re.search(r'(stratum)=([^,]+)', raw_peer_data)
  311. if not stratum:
  312. continue
  313. if int(stratum.group(2)) > 15:
  314. continue
  315. new_peer = Peer(idx=peer_id, name=srcadr)
  316. new_peers.append(new_peer)
  317. return new_peers
  318. def get_peer_ids(self, res):
  319. """
  320. Unpack the NTP Control Message header
  321. Get data length from header
  322. Get list of association ids returned in the readstat response
  323. """
  324. try:
  325. count = struct.unpack(HEADER_FORMAT, res[:HEADER_LEN])[6]
  326. except struct.error as error:
  327. self.error('error unpacking header: {0}'.format(error))
  328. return None
  329. if not count:
  330. self.error('empty data field in NTP control packet')
  331. return None
  332. data_end = HEADER_LEN + count
  333. data = res[HEADER_LEN:data_end]
  334. data_format = ''.join(['!', 'H' * int(count / 2)])
  335. try:
  336. peer_ids = list(struct.unpack(data_format, data))[::2]
  337. except struct.error as error:
  338. self.error('error unpacking data: {0}'.format(error))
  339. return None
  340. return peer_ids