|
@@ -12,6 +12,10 @@ logger = logging.getLogger(ROOT_LOGGER_NAME).getChild(__name__)
|
12
|
12
|
|
13
|
13
|
|
14
|
14
|
class base(object):
|
|
15
|
+ MONITORING_HEARTBEAT = "heartbeat"
|
|
16
|
+ MONITORING_BATTERY = "battery"
|
|
17
|
+ MONITORING_FOLLOW_SETPOINT = "follow_setpoint"
|
|
18
|
+
|
15
|
19
|
FOLLOW_REQUEST_WARNING = 5
|
16
|
20
|
FOLLOW_REQUEST_ERROR = 60
|
17
|
21
|
FOLLOW_KEY = None
|
|
@@ -19,12 +23,14 @@ class base(object):
|
19
|
23
|
BATTERY_LVL_WARNING = 15
|
20
|
24
|
BATTERY_LVL_ERROR = 5
|
21
|
25
|
|
22
|
|
- LAST_MSG_WARNING = 6 * 24 * 60 * 60
|
23
|
|
- LAST_MSG_ERROR = 24 * 24 * 60 * 60
|
|
26
|
+ LAST_MSG_WARNING = 6 * 60 * 60
|
|
27
|
+ LAST_MSG_ERROR = 24 * 60 * 60
|
24
|
28
|
|
25
|
29
|
def __init__(self, mqtt_client: mqtt.mqtt_client, topic):
|
26
|
30
|
self.topic = topic
|
27
|
31
|
|
|
32
|
+ self.__unknown_tm__ = {}
|
|
33
|
+
|
28
|
34
|
mqtt_client.add_callback(topic, self.__rx__)
|
29
|
35
|
mqtt_client.add_callback(topic + '/#', self.__rx__)
|
30
|
36
|
|
|
@@ -58,8 +64,8 @@ class base(object):
|
58
|
64
|
|
59
|
65
|
|
60
|
66
|
|
61
|
|
- if "battery" in payload and message.topic == self.topic:
|
62
|
|
- self.battery = payload["battery"]
|
|
67
|
+ if self.MONITORING_BATTERY in payload and message.topic == self.topic:
|
|
68
|
+ self.battery = payload[self.MONITORING_BATTERY]
|
63
|
69
|
|
64
|
70
|
def target(self, key, value):
|
65
|
71
|
tm_t, value_t = self.__target_storage__.get(key, (0, None))
|
|
@@ -75,51 +81,70 @@ class base(object):
|
75
|
81
|
|
76
|
82
|
|
77
|
83
|
|
78
|
|
- if key == "heartbeat":
|
|
84
|
+ if key == self.MONITORING_HEARTBEAT:
|
79
|
85
|
if self.last_device_msg is None:
|
80
|
|
- return {"status": nagios.Nagios.UNKNOWN, "msg": "Device exists, but no data received or unknown monitoring"}
|
|
86
|
+ return self.__nagios_return__(self.MONITORING_HEARTBEAT, nagios.Nagios.UNKNOWN, "Device exists, but no data received")
|
81
|
87
|
else:
|
82
|
88
|
dt = time.time() - self.last_device_msg
|
83
|
89
|
dt_disp = dt / 60 / 60
|
84
|
90
|
if dt > self.LAST_MSG_ERROR:
|
85
|
|
- return {"status": nagios.Nagios.ERROR, "msg": "Last message %.1fh ago" % dt_disp}
|
|
91
|
+ return self.__nagios_return__(self.MONITORING_HEARTBEAT, nagios.Nagios.ERROR, "Last message %.1fh ago" % dt_disp)
|
86
|
92
|
elif dt > self.LAST_MSG_WARNING:
|
87
|
|
- return {"status": nagios.Nagios.WARNING, "msg": "Last message %.1fh ago" % dt_disp}
|
|
93
|
+ return self.__nagios_return__(self.MONITORING_HEARTBEAT, nagios.Nagios.WARNING, "Last message %.1fh ago" % dt_disp)
|
88
|
94
|
else:
|
89
|
|
- return {"status": nagios.Nagios.OK, "msg": "Last message %.1fh ago" % dt_disp}
|
|
95
|
+ return self.__nagios_return__(self.MONITORING_HEARTBEAT, nagios.Nagios.OK, "Last message %.1fh ago" % dt_disp)
|
90
|
96
|
|
91
|
97
|
|
92
|
98
|
|
93
|
|
- elif key == 'follow_setpoint':
|
|
99
|
+ elif key == self.MONITORING_FOLLOW_SETPOINT:
|
94
|
100
|
if self.FOLLOW_KEY is None:
|
95
|
|
- return {"status": nagios.Nagios.UNKNOWN, "msg": "Device exist, but does not follow any setpoint."}
|
|
101
|
+ return self.__nagios_return__(self.MONITORING_FOLLOW_SETPOINT, nagios.Nagios.UNKNOWN, "Device exist, but does not follow any setpoint.", force=True)
|
96
|
102
|
tm_s, value_s = self.__state_storage__.get(self.FOLLOW_KEY, (0, None))
|
97
|
103
|
try:
|
98
|
104
|
tm_t, value_t = self.__target_storage__[self.FOLLOW_KEY]
|
99
|
105
|
except KeyError:
|
100
|
106
|
if value_s is not None:
|
101
|
|
- return {"status": nagios.Nagios.WARNING, "msg": "Current temperature setpoint %.1f°C (age=%.1fmin), but never received a setpoint" % (value_s, (time.time()-tm_s)/60)}
|
102
|
|
- return {"status": nagios.Nagios.UNKNOWN, "msg": "Device exists, but no data received"}
|
|
107
|
+ return self.__nagios_return__(self.MONITORING_FOLLOW_SETPOINT, nagios.Nagios.WARNING, "Current temperature setpoint %.1f°C (age=%.1fmin), but never received a setpoint" % (value_s, (time.time()-tm_s)/60))
|
|
108
|
+ return self.__nagios_return__(self.MONITORING_FOLLOW_SETPOINT, nagios.Nagios.UNKNOWN, "Device exists, but no data received")
|
103
|
109
|
else:
|
104
|
110
|
tm = time.time()
|
105
|
111
|
dt = tm - tm_t
|
106
|
112
|
if value_t != value_s and dt > self.FOLLOW_REQUEST_ERROR:
|
107
|
|
- return {"status": nagios.Nagios.ERROR, "msg": "Requested setpoint unequal valve setpoint %.1f°C since %.1fmin" % (value_s, (time.time()-tm_s)/60)}
|
|
113
|
+ return self.__nagios_return__(self.MONITORING_FOLLOW_SETPOINT, nagios.Nagios.ERROR, "Requested setpoint unequal valve setpoint %.1f°C since %.1fmin" % (value_s, (time.time()-tm_s)/60))
|
108
|
114
|
elif value_t != value_s and dt > self.FOLLOW_REQUEST_WARNING:
|
109
|
|
- return {"status": nagios.Nagios.WARNING, "msg": "Requested setpoint unequal valve setpoint %.1f°C since %.1fmin" % (value_s, (time.time()-tm_s))}
|
110
|
|
- return {"status": nagios.Nagios.OK, "msg": "Requested setpoint equal valve setpoint %.1f°C" % value_s}
|
|
115
|
+ return self.__nagios_return__(self.MONITORING_FOLLOW_SETPOINT, nagios.Nagios.WARNING, "Requested setpoint unequal valve setpoint %.1f°C since %.1fmin" % (value_s, (time.time()-tm_s)))
|
|
116
|
+ return self.__nagios_return__(self.MONITORING_FOLLOW_SETPOINT, nagios.Nagios.OK, "Requested setpoint equal valve setpoint %.1f°C" % value_s)
|
111
|
117
|
|
112
|
118
|
|
113
|
119
|
|
114
|
|
- elif key == "battery":
|
|
120
|
+ elif key == self.MONITORING_BATTERY:
|
115
|
121
|
if self.battery is None:
|
116
|
|
- return {"status": nagios.Nagios.UNKNOWN, "msg": "Device exists, but no data received or unknown monitoring"}
|
|
122
|
+ return self.__nagios_return__(self.MONITORING_BATTERY, nagios.Nagios.UNKNOWN, "Device exists, but no data received or unknown monitoring")
|
117
|
123
|
elif self.battery <= self.BATTERY_LVL_ERROR:
|
118
|
|
- return {"status": nagios.Nagios.ERROR, "msg": "Battery level critical low (%.1f%%)" % self.battery}
|
|
124
|
+ return self.__nagios_return__(self.MONITORING_BATTERY, nagios.Nagios.ERROR, "Battery level critical low (%.1f%%)" % self.battery)
|
119
|
125
|
elif self.battery <= self.BATTERY_LVL_WARNING:
|
120
|
|
- return {"status": nagios.Nagios.WARNING, "msg": "Battery level low (%.1f%%)" % self.battery}
|
|
126
|
+ return self.__nagios_return__(self.MONITORING_BATTERY, nagios.Nagios.WARNING, "Battery level low (%.1f%%)" % self.battery)
|
121
|
127
|
else:
|
122
|
|
- return {"status": nagios.Nagios.OK, "msg": "Battery okay (%.1f%%)" % self.battery}
|
|
128
|
+ return self.__nagios_return__(self.MONITORING_BATTERY, nagios.Nagios.OK, "Battery okay (%.1f%%)" % self.battery)
|
|
129
|
+
|
|
130
|
+ def __nagios_return__(self, monitoring_name, status, msg, force=False):
|
|
131
|
+ tm = time.time()
|
|
132
|
+ if monitoring_name not in self.__unknown_tm__:
|
|
133
|
+ self.__unknown_tm__[monitoring_name] = None
|
|
134
|
+ if status == nagios.Nagios.UNKNOWN and not force:
|
|
135
|
+ if self.__unknown_tm__[monitoring_name] is None:
|
|
136
|
+ self.__unknown_tm__[monitoring_name] = tm
|
|
137
|
+ dt = tm - self.__unknown_tm__[monitoring_name]
|
|
138
|
+ if dt >= self.LAST_MSG_ERROR:
|
|
139
|
+ status = nagios.Nagios.UNKNOWN
|
|
140
|
+ elif dt >= self.LAST_MSG_WARNING:
|
|
141
|
+ status = nagios.Nagios.WARNING
|
|
142
|
+ else:
|
|
143
|
+ status = nagios.Nagios.OK
|
|
144
|
+ msg += " - since %.1fh" % (dt / 3600)
|
|
145
|
+ else:
|
|
146
|
+ self.__unknown_tm__[monitoring_name] = None
|
|
147
|
+ return {"status": status, "msg": msg}
|
123
|
148
|
|
124
|
149
|
|
125
|
150
|
class group(object):
|