-
Notifications
You must be signed in to change notification settings - Fork 60
/
prometheus.js
119 lines (107 loc) · 5.04 KB
/
prometheus.js
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
/*
* This script exposes a "/status" endpoint that returns Prometheus metrics that can be scraped.
* It will be reachable under "<ip>/script/<id>/metrics". Id will be 1 if this is your first script.
*
* Example Prometheus config:
*
* scrape_configs:
* - job_name: 'shelly'
* metrics_path: /script/1/metrics
* static_configs:
* - targets: ['<ip>']
*
* Replace <ip> with the ip address of the device and adjust the script id if needed.
*
* If you use Grafana to display metrics scraped, you can give a try to the example dashboard in prometheus-grafana-example-dashboard.json
* available via the url https://raw.githubusercontent.com/ALLTERCO/shelly-script-examples/main/prometheus-grafana-example-dashboard.json
*
* Note: This script assumes you have one switch, but you can configure that in the Configuration section
*/
// ---------------------------------------------------------------------------------------------------------------------
// Configuration (you can change / adapt here)
// ---------------------------------------------------------------------------------------------------------------------
// Prefix added to all metrics name
const metric_prefix = "shelly_"
// url of metrics. will be used for the last part of "<ip>/script/<id>/<url>"
// where ip = your plug ip/hostname; id the script id or number (1 if this is the first); url the value below
const url = "metrics"
// List of internal switch to monitor using the shelly's id naming scheme in the form of switch:<id> (ex switch:0)
const monitored_switches = ["switch:0"]
// ---------------------------------------------------------------------------------------------------------------------
// Prometheus exporter
// ---------------------------------------------------------------------------------------------------------------------
const TYPE_GAUGE = "gauge"
const TYPE_COUNTER = "counter"
var info = Shelly.getDeviceInfo();
function promLabel(label, value) {
return [label, "=", '"', value, '"'].join("");
}
// Will be added to every metrics
var defaultLabels = [
["name", info.name],
["id", info.id],
["mac", info.mac],
["app", info.app]
]
.map(function (data) {
return promLabel(data[0], data[1]);
});
/**
* Generate one metric output with all the shenanigans around it
* @param name The name of the metrics (Will be prefixed by metric_prefix value)
* @param type One of the TYPE_* of metrics. Usually Counter, Gauge, Histogram (not supported yes)
* @param specificLabels Array of labels generated by promLabel() specific to this metric.
* @param description An Human description of the metric
* @param value The actual metric numeric value
* @returns {string} The metric string to include in final response
*/
function printPrometheusMetric(name, type, specificLabels, description, value) {
return [
"# HELP ", metric_prefix, name, " ", description, "\n",
"# HELP ", metric_prefix, name, " ", type, "\n",
metric_prefix, name, "{", defaultLabels.join(","), specificLabels.length > 0 ? "," : "", specificLabels.join(","), "}", " ", value, "\n\n"
].join("");
}
/**
* HTTP handler that will be called when the url will be accessed
* @param request
* @param response
*/
function httpServerHandler(request, response) {
response.body = [
generateMetricsForSystem(),
monitored_switches.map(function (switch_string_id) {
return generateMetricsForSwitch(switch_string_id)
}).join("")
].join("")
response.code = 200;
response.send();
}
/**
* Generate metrics for the system part
* @returns {string}
*/
function generateMetricsForSystem() {
const sys = Shelly.getComponentStatus("sys")
return [
printPrometheusMetric("uptime_seconds", TYPE_COUNTER, [], "power level in watts", sys.uptime),
printPrometheusMetric("ram_size_bytes", TYPE_GAUGE, [], "Internal board RAM size in bytes", sys.ram_size),
printPrometheusMetric("ram_free_bytes", TYPE_GAUGE, [], "Internal board free RAM size in bytes", sys.ram_free)
].join("")
}
/**
* generate metrics for one switch with the name given as parameter
* @returns {string}
*/
function generateMetricsForSwitch(string_id) {
const sw = Shelly.getComponentStatus(string_id);
return [
printPrometheusMetric("switch_power_watts", TYPE_GAUGE, [promLabel("switch", sw.id)], "Instant power consumption in watts", sw.apower),
printPrometheusMetric("switch_voltage_volts", TYPE_GAUGE, [promLabel("switch", sw.id)], "Instant voltage in volts", sw.voltage),
printPrometheusMetric("switch_current_amperes", TYPE_GAUGE, [promLabel("switch", sw.id)], "Instant current in amperes", sw.current),
printPrometheusMetric("switch_temperature_celsius", TYPE_GAUGE, [promLabel("switch", sw.id)], "temperature of the plug in celsius", sw.temperature.tC),
printPrometheusMetric("switch_power_total", TYPE_COUNTER, [promLabel("switch", sw.id)], "accumulated energy consumed in watts hours", sw.aenergy.total),
printPrometheusMetric("switch_output", TYPE_GAUGE, [promLabel("switch", sw.id)], "is switch (1)on or (0)off", sw.output ? 1 : 0)
].join("");
}
HTTPServer.registerEndpoint(url, httpServerHandler);