freeradius.chart.py 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. # -*- coding: utf-8 -*-
  2. # Description: freeradius netdata python.d module
  3. # Author: ilyam8
  4. # SPDX-License-Identifier: GPL-3.0-or-later
  5. import re
  6. from subprocess import Popen, PIPE
  7. from bases.FrameworkServices.SimpleService import SimpleService
  8. from bases.collection import find_binary
  9. update_every = 15
  10. PARSER = re.compile(r'((?<=-)[AP][a-zA-Z-]+) = (\d+)')
  11. RADIUS_MSG = 'Message-Authenticator = 0x00, FreeRADIUS-Statistics-Type = 15, Response-Packet-Type = Access-Accept'
  12. RADCLIENT_RETRIES = 1
  13. RADCLIENT_TIMEOUT = 1
  14. DEFAULT_HOST = 'localhost'
  15. DEFAULT_PORT = 18121
  16. DEFAULT_DO_ACCT = False
  17. DEFAULT_DO_PROXY_AUTH = False
  18. DEFAULT_DO_PROXY_ACCT = False
  19. ORDER = [
  20. 'authentication',
  21. 'accounting',
  22. 'proxy-auth',
  23. 'proxy-acct',
  24. ]
  25. CHARTS = {
  26. 'authentication': {
  27. 'options': [None, 'Authentication', 'packets/s', 'authentication', 'freerad.auth', 'line'],
  28. 'lines': [
  29. ['access-accepts', None, 'incremental'],
  30. ['access-rejects', None, 'incremental'],
  31. ['auth-dropped-requests', 'dropped-requests', 'incremental'],
  32. ['auth-duplicate-requests', 'duplicate-requests', 'incremental'],
  33. ['auth-invalid-requests', 'invalid-requests', 'incremental'],
  34. ['auth-malformed-requests', 'malformed-requests', 'incremental'],
  35. ['auth-unknown-types', 'unknown-types', 'incremental']
  36. ]
  37. },
  38. 'accounting': {
  39. 'options': [None, 'Accounting', 'packets/s', 'accounting', 'freerad.acct', 'line'],
  40. 'lines': [
  41. ['accounting-requests', 'requests', 'incremental'],
  42. ['accounting-responses', 'responses', 'incremental'],
  43. ['acct-dropped-requests', 'dropped-requests', 'incremental'],
  44. ['acct-duplicate-requests', 'duplicate-requests', 'incremental'],
  45. ['acct-invalid-requests', 'invalid-requests', 'incremental'],
  46. ['acct-malformed-requests', 'malformed-requests', 'incremental'],
  47. ['acct-unknown-types', 'unknown-types', 'incremental']
  48. ]
  49. },
  50. 'proxy-auth': {
  51. 'options': [None, 'Proxy Authentication', 'packets/s', 'authentication', 'freerad.proxy.auth', 'line'],
  52. 'lines': [
  53. ['proxy-access-accepts', 'access-accepts', 'incremental'],
  54. ['proxy-access-rejects', 'access-rejects', 'incremental'],
  55. ['proxy-auth-dropped-requests', 'dropped-requests', 'incremental'],
  56. ['proxy-auth-duplicate-requests', 'duplicate-requests', 'incremental'],
  57. ['proxy-auth-invalid-requests', 'invalid-requests', 'incremental'],
  58. ['proxy-auth-malformed-requests', 'malformed-requests', 'incremental'],
  59. ['proxy-auth-unknown-types', 'unknown-types', 'incremental']
  60. ]
  61. },
  62. 'proxy-acct': {
  63. 'options': [None, 'Proxy Accounting', 'packets/s', 'accounting', 'freerad.proxy.acct', 'line'],
  64. 'lines': [
  65. ['proxy-accounting-requests', 'requests', 'incremental'],
  66. ['proxy-accounting-responses', 'responses', 'incremental'],
  67. ['proxy-acct-dropped-requests', 'dropped-requests', 'incremental'],
  68. ['proxy-acct-duplicate-requests', 'duplicate-requests', 'incremental'],
  69. ['proxy-acct-invalid-requests', 'invalid-requests', 'incremental'],
  70. ['proxy-acct-malformed-requests', 'malformed-requests', 'incremental'],
  71. ['proxy-acct-unknown-types', 'unknown-types', 'incremental']
  72. ]
  73. }
  74. }
  75. def radclient_status(radclient, retries, timeout, host, port, secret):
  76. # radclient -r 1 -t 1 -x 127.0.0.1:18121 status secret
  77. return '{radclient} -r {num_retries} -t {timeout} -x {host}:{port} status {secret}'.format(
  78. radclient=radclient,
  79. num_retries=retries,
  80. timeout=timeout,
  81. host=host,
  82. port=port,
  83. secret=secret,
  84. ).split()
  85. class Service(SimpleService):
  86. def __init__(self, configuration=None, name=None):
  87. SimpleService.__init__(self, configuration=configuration, name=name)
  88. self.order = ORDER
  89. self.definitions = CHARTS
  90. self.host = self.configuration.get('host', DEFAULT_HOST)
  91. self.port = self.configuration.get('port', DEFAULT_PORT)
  92. self.secret = self.configuration.get('secret')
  93. self.do_acct = self.configuration.get('acct', DEFAULT_DO_ACCT)
  94. self.do_proxy_auth = self.configuration.get('proxy_auth', DEFAULT_DO_PROXY_AUTH)
  95. self.do_proxy_acct = self.configuration.get('proxy_acct', DEFAULT_DO_PROXY_ACCT)
  96. self.echo = find_binary('echo')
  97. self.radclient = find_binary('radclient')
  98. self.sub_echo = [self.echo, RADIUS_MSG]
  99. self.sub_radclient = radclient_status(
  100. self.radclient, RADCLIENT_RETRIES, RADCLIENT_TIMEOUT, self.host, self.port, self.secret,
  101. )
  102. def check(self):
  103. if not self.radclient:
  104. self.error("Can't locate 'radclient' binary or binary is not executable by netdata user")
  105. return False
  106. if not self.echo:
  107. self.error("Can't locate 'echo' binary or binary is not executable by netdata user")
  108. return None
  109. if not self.secret:
  110. self.error("'secret' isn't set")
  111. return None
  112. if not self.get_raw_data():
  113. self.error('Request returned no data. Is server alive?')
  114. return False
  115. if not self.do_acct:
  116. self.order.remove('accounting')
  117. if not self.do_proxy_auth:
  118. self.order.remove('proxy-auth')
  119. if not self.do_proxy_acct:
  120. self.order.remove('proxy-acct')
  121. return True
  122. def get_data(self):
  123. """
  124. Format data received from shell command
  125. :return: dict
  126. """
  127. result = self.get_raw_data()
  128. if not result:
  129. return None
  130. return dict(
  131. (key.lower(), value) for key, value in PARSER.findall(result)
  132. )
  133. def get_raw_data(self):
  134. """
  135. The following code is equivalent to
  136. 'echo "Message-Authenticator = 0x00, FreeRADIUS-Statistics-Type = 15, Response-Packet-Type = Access-Accept"
  137. | radclient -t 1 -r 1 host:port status secret'
  138. :return: str
  139. """
  140. try:
  141. process_echo = Popen(self.sub_echo, stdout=PIPE, stderr=PIPE, shell=False)
  142. process_rad = Popen(self.sub_radclient, stdin=process_echo.stdout, stdout=PIPE, stderr=PIPE, shell=False)
  143. process_echo.stdout.close()
  144. raw_result = process_rad.communicate()[0]
  145. except OSError:
  146. return None
  147. if process_rad.returncode is 0:
  148. return raw_result.decode()
  149. return None