redis.chart.py 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. # -*- coding: utf-8 -*-
  2. # Description: redis netdata python.d module
  3. # Author: Pawel Krupa (paulfantom)
  4. from bases.FrameworkServices.SocketService import SocketService
  5. # default module values (can be overridden per job in `config`)
  6. priority = 60000
  7. retries = 60
  8. # default job configuration (overridden by python.d.plugin)
  9. # config = {'local': {
  10. # 'update_every': update_every,
  11. # 'retries': retries,
  12. # 'priority': priority,
  13. # 'host': 'localhost',
  14. # 'port': 6379,
  15. # 'unix_socket': None
  16. # }}
  17. ORDER = ['operations', 'hit_rate', 'memory', 'keys', 'net', 'connections', 'clients', 'slaves', 'persistence',
  18. 'bgsave_now', 'bgsave_health']
  19. CHARTS = {
  20. 'operations': {
  21. 'options': [None, 'Redis Operations', 'operations/s', 'operations', 'redis.operations', 'line'],
  22. 'lines': [
  23. ['total_commands_processed', 'commands', 'incremental'],
  24. ['instantaneous_ops_per_sec', 'operations', 'absolute']
  25. ]},
  26. 'hit_rate': {
  27. 'options': [None, 'Redis Hit rate', 'percent', 'hits', 'redis.hit_rate', 'line'],
  28. 'lines': [
  29. ['hit_rate', 'rate', 'absolute']
  30. ]},
  31. 'memory': {
  32. 'options': [None, 'Redis Memory utilization', 'kilobytes', 'memory', 'redis.memory', 'line'],
  33. 'lines': [
  34. ['used_memory', 'total', 'absolute', 1, 1024],
  35. ['used_memory_lua', 'lua', 'absolute', 1, 1024]
  36. ]},
  37. 'net': {
  38. 'options': [None, 'Redis Bandwidth', 'kilobits/s', 'network', 'redis.net', 'area'],
  39. 'lines': [
  40. ['total_net_input_bytes', 'in', 'incremental', 8, 1024],
  41. ['total_net_output_bytes', 'out', 'incremental', -8, 1024]
  42. ]},
  43. 'keys': {
  44. 'options': [None, 'Redis Keys per Database', 'keys', 'keys', 'redis.keys', 'line'],
  45. 'lines': [
  46. # lines are created dynamically in `check()` method
  47. ]},
  48. 'connections': {
  49. 'options': [None, 'Redis Connections', 'connections/s', 'connections', 'redis.connections', 'line'],
  50. 'lines': [
  51. ['total_connections_received', 'received', 'incremental', 1],
  52. ['rejected_connections', 'rejected', 'incremental', -1]
  53. ]},
  54. 'clients': {
  55. 'options': [None, 'Redis Clients', 'clients', 'connections', 'redis.clients', 'line'],
  56. 'lines': [
  57. ['connected_clients', 'connected', 'absolute', 1],
  58. ['blocked_clients', 'blocked', 'absolute', -1]
  59. ]},
  60. 'slaves': {
  61. 'options': [None, 'Redis Slaves', 'slaves', 'replication', 'redis.slaves', 'line'],
  62. 'lines': [
  63. ['connected_slaves', 'connected', 'absolute']
  64. ]},
  65. 'persistence': {
  66. 'options': [None, 'Redis Persistence Changes Since Last Save', 'changes', 'persistence',
  67. 'redis.rdb_changes', 'line'],
  68. 'lines': [
  69. ['rdb_changes_since_last_save', 'changes', 'absolute']
  70. ]},
  71. 'bgsave_now': {
  72. 'options': [None, 'Duration of the RDB Save Operation', 'seconds', 'persistence',
  73. 'redis.bgsave_now', 'absolute'],
  74. 'lines': [
  75. ['rdb_bgsave_in_progress', 'rdb save', 'absolute']
  76. ]},
  77. 'bgsave_health': {
  78. 'options': [None, 'Status of the Last RDB Save Operation', 'status', 'persistence',
  79. 'redis.bgsave_health', 'line'],
  80. 'lines': [
  81. ['rdb_last_bgsave_status', 'rdb save', 'absolute']
  82. ]}
  83. }
  84. class Service(SocketService):
  85. def __init__(self, configuration=None, name=None):
  86. SocketService.__init__(self, configuration=configuration, name=name)
  87. self.order = ORDER
  88. self.definitions = CHARTS
  89. self._keep_alive = True
  90. self.chart_name = ""
  91. self.host = self.configuration.get('host', 'localhost')
  92. self.port = self.configuration.get('port', 6379)
  93. self.unix_socket = self.configuration.get('socket')
  94. password = self.configuration.get('pass', str())
  95. self.bgsave_time = 0
  96. self.requests = dict(request='INFO\r\n'.encode(),
  97. password=' '.join(['AUTH', password, '\r\n']).encode() if password else None)
  98. self.request = self.requests['request']
  99. def _get_data(self):
  100. """
  101. Get data from socket
  102. :return: dict
  103. """
  104. if self.requests['password']:
  105. self.request = self.requests['password']
  106. raw = self._get_raw_data().strip()
  107. if raw != "+OK":
  108. self.error("invalid password")
  109. return None
  110. self.request = self.requests['request']
  111. response = self._get_raw_data()
  112. if response is None:
  113. # error has already been logged
  114. return None
  115. try:
  116. parsed = response.split("\n")
  117. except AttributeError:
  118. self.error("response is invalid/empty")
  119. return None
  120. data = dict()
  121. for line in parsed:
  122. if len(line) < 5 or line[0] == '$' or line[0] == '#':
  123. continue
  124. if line.startswith('db'):
  125. tmp = line.split(',')[0].replace('keys=', '')
  126. record = tmp.split(':')
  127. data[record[0]] = record[1]
  128. continue
  129. try:
  130. t = line.split(':')
  131. data[t[0]] = t[1]
  132. except (IndexError, ValueError):
  133. self.debug("invalid line received: " + str(line))
  134. if not data:
  135. self.error("received data doesn't have any records")
  136. return None
  137. try:
  138. data['hit_rate'] = (int(data['keyspace_hits']) * 100) / (int(data['keyspace_hits'])
  139. + int(data['keyspace_misses']))
  140. except (KeyError, ZeroDivisionError, TypeError):
  141. data['hit_rate'] = 0
  142. if data['rdb_bgsave_in_progress'] != '0\r':
  143. self.bgsave_time += self.update_every
  144. else:
  145. self.bgsave_time = 0
  146. data['rdb_last_bgsave_status'] = 0 if data['rdb_last_bgsave_status'] == 'ok\r' else 1
  147. data['rdb_bgsave_in_progress'] = self.bgsave_time
  148. return data
  149. def _check_raw_data(self, data):
  150. """
  151. Check if all data has been gathered from socket.
  152. Parse first line containing message length and check against received message
  153. :param data: str
  154. :return: boolean
  155. """
  156. length = len(data)
  157. supposed = data.split('\n')[0][1:-1]
  158. offset = len(supposed) + 4 # 1 dollar sing, 1 new line character + 1 ending sequence '\r\n'
  159. if not supposed.isdigit():
  160. return True
  161. supposed = int(supposed)
  162. if length - offset >= supposed:
  163. self.debug("received full response from redis")
  164. return True
  165. self.debug("waiting more data from redis")
  166. return False
  167. def check(self):
  168. """
  169. Parse configuration, check if redis is available, and dynamically create chart lines data
  170. :return: boolean
  171. """
  172. data = self._get_data()
  173. if data is None:
  174. return False
  175. for name in data:
  176. if name.startswith('db'):
  177. self.definitions['keys']['lines'].append([name, None, 'absolute'])
  178. return True