forked from FergusInLondon/Tapo-P110-Prometheus-Exporter
-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathcollector.py
160 lines (130 loc) · 5.43 KB
/
collector.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
from contextlib import contextmanager
from enum import Enum, auto
from math import floor
from time import time
from time import sleep
from loguru import logger
from prometheus_client import Histogram
from prometheus_client.core import GaugeMetricFamily
from PyP100 import PyP110
OBSERVATION_RED_METRICS = Histogram(
"tapo_p110_observation_rate_ms",
"RED metrics for queries to the TP-Link TAPO P110 devices. (milliseconds)",
labelnames=["ip_address", "room", "success"],
buckets=(10, 100, 150, 200, 250, 300, 500, 750, 1000, 1500, 2000)
)
class MetricType(Enum):
DEVICE_COUNT = auto()
TODAY_RUNTIME = auto()
MONTH_RUNTIME = auto()
TODAY_ENERGY = auto()
MONTH_ENERGY = auto()
CURRENT_POWER = auto()
def get_metrics():
return {
MetricType.DEVICE_COUNT: GaugeMetricFamily(
"tapo_p110_device_count",
"Number of available TP-Link TAPO P110 Smart Sockets.",
),
MetricType.TODAY_RUNTIME: GaugeMetricFamily(
"tapo_p110_today_runtime_mins",
"Current running time for the TP-Link TAPO P110 Smart Socket today. (minutes)",
labels=["ip_address", "room"],
),
MetricType.MONTH_RUNTIME: GaugeMetricFamily(
"tapo_p110_month_runtime_mins",
"Current running time for the TP-Link TAPO P110 Smart Socket this month. (minutes)",
labels=["ip_address", "room"],
),
MetricType.TODAY_ENERGY: GaugeMetricFamily(
"tapo_p110_today_energy_wh",
"Energy consumed by the TP-Link TAPO P110 Smart Socket today. (Watt-hours)",
labels=["ip_address", "room"],
),
MetricType.MONTH_ENERGY: GaugeMetricFamily(
"tapo_p110_month_energy_wh",
"Energy consumed by the TP-Link TAPO P110 Smart Socket this month. (Watt-hours)",
labels=["ip_address", "room"],
),
MetricType.CURRENT_POWER: GaugeMetricFamily(
"tapo_p110_power_consumption_w",
"Current power consumption for TP-Link TAPO P110 Smart Socket. (Watts)",
labels=["ip_address", "room"],
),
}
RED_SUCCESS = "SUCCESS"
RED_FAILURE = "FAILURE"
@contextmanager
def time_observation(ip_address, room):
caught = None
status = RED_SUCCESS
start = time()
try:
yield
except Exception as e:
status = RED_FAILURE
caught = e
duration = floor((time() - start) * 1000)
OBSERVATION_RED_METRICS.labels(ip_address=ip_address, room=room, success=status).observe(duration)
logger.debug("observation completed", extra={
"ip": ip_address, "room": room, "duration_ms": duration,
})
if caught:
raise caught
class Collector:
def __init__(self, deviceMap, email_address, password):
def create_device(ip_address, room):
extra = {
"ip": ip_address,
"room": room,
}
logger.debug("connecting to device", extra=extra)
exception_count = 0 # Counter for exceptions
while True:
try:
d = PyP110.P110(ip_address, email_address, password)
d.handshake()
d.login()
except Exception as e:
exception_count += 1
logger.error("failed to connect to device", extra=extra, exc_info=True)
if exception_count >= 3: # Return None after the third exception
d = None
break
sleep(1) # Sleep for 1 second after each exception
continue
break
if d is not None:
logger.debug("successfully authenticated with device", extra=extra)
return d
self.devices = {
room: (ip_address, device)
for room, ip_address in deviceMap.items()
if (device := create_device(ip_address, room)) is not None
}
def get_device_data(self, device, ip_address, room):
with time_observation(ip_address, room):
logger.debug("retrieving energy usage statistics for device", extra={
"ip": ip_address, "room": room,
})
return device.getEnergyUsage()
def collect(self):
logger.info("recieving prometheus metrics scrape: collecting observations")
metrics = get_metrics()
metrics[MetricType.DEVICE_COUNT].add_metric([], len(self.devices))
for room, (ip_addr, device) in self.devices.items():
logger.info("performing observations for device", extra={
"ip": ip_addr, "room": room,
})
try:
data = self.get_device_data(device, ip_addr, room)
labels = [ip_addr, room]
metrics[MetricType.TODAY_RUNTIME].add_metric(labels, data['today_runtime'])
metrics[MetricType.MONTH_RUNTIME].add_metric(labels, data['month_runtime'])
metrics[MetricType.TODAY_ENERGY].add_metric(labels, data['today_energy'])
metrics[MetricType.MONTH_ENERGY].add_metric(labels, data['month_energy'])
metrics[MetricType.CURRENT_POWER].add_metric(labels, data['current_power'])
except Exception as e:
logger.exception("encountered exception during observation!")
for m in metrics.values():
yield m