2024-08-24 18:02:57 +02:00
|
|
|
const { R } = require("redbean-node");
|
2021-08-10 11:51:30 +02:00
|
|
|
const PrometheusClient = require("prom-client");
|
2022-04-13 17:33:37 +02:00
|
|
|
const { log } = require("../src/util");
|
2021-07-27 18:52:31 +02:00
|
|
|
|
|
|
|
const commonLabels = [
|
2021-08-10 11:51:30 +02:00
|
|
|
"monitor_name",
|
|
|
|
"monitor_type",
|
|
|
|
"monitor_url",
|
|
|
|
"monitor_hostname",
|
|
|
|
"monitor_port",
|
2021-10-10 10:37:53 +02:00
|
|
|
];
|
2021-07-27 18:52:31 +02:00
|
|
|
|
|
|
|
class Prometheus {
|
|
|
|
|
2022-04-20 20:56:40 +02:00
|
|
|
/**
|
2024-08-24 18:02:57 +02:00
|
|
|
* Metric: monitor_cert_days_remaining
|
|
|
|
* @type {PrometheusClient.Gauge<string> | null}
|
|
|
|
*/
|
|
|
|
static monitorCertDaysRemaining = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Metric: monitor_cert_is_valid
|
|
|
|
* @type {PrometheusClient.Gauge<string> | null}
|
|
|
|
*/
|
|
|
|
static monitorCertIsValid = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Metric: monitor_response_time
|
|
|
|
* @type {PrometheusClient.Gauge<string> | null}
|
|
|
|
*/
|
|
|
|
static monitorResponseTime = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Metric: monitor_status
|
|
|
|
* @type {PrometheusClient.Gauge<string> | null}
|
|
|
|
*/
|
|
|
|
static monitorStatus = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* All registered metric labels.
|
|
|
|
* @type {string[] | null}
|
|
|
|
*/
|
|
|
|
static monitorLabelNames = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Monitor labels/values combination.
|
|
|
|
* @type {{}}
|
|
|
|
*/
|
|
|
|
monitorLabelValues;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialize metrics and get all label names the first time called.
|
|
|
|
* @returns {void}
|
2022-04-20 20:56:40 +02:00
|
|
|
*/
|
2024-08-24 18:02:57 +02:00
|
|
|
static async initMetrics() {
|
|
|
|
if (!this.monitorLabelNames) {
|
|
|
|
let labelNames = await R.getCol("SELECT name FROM tag");
|
|
|
|
this.monitorLabelNames = [ ...commonLabels, ...labelNames ];
|
|
|
|
}
|
|
|
|
if (!this.monitorCertDaysRemaining) {
|
|
|
|
this.monitorCertDaysRemaining = new PrometheusClient.Gauge({
|
|
|
|
name: "monitor_cert_days_remaining",
|
|
|
|
help: "The number of days remaining until the certificate expires",
|
|
|
|
labelNames: this.monitorLabelNames
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (!this.monitorCertIsValid) {
|
|
|
|
this.monitorCertIsValid = new PrometheusClient.Gauge({
|
|
|
|
name: "monitor_cert_is_valid",
|
|
|
|
help: "Is the certificate still valid? (1 = Yes, 0 = No)",
|
|
|
|
labelNames: this.monitorLabelNames
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (!this.monitorResponseTime) {
|
|
|
|
this.monitorResponseTime = new PrometheusClient.Gauge({
|
|
|
|
name: "monitor_response_time",
|
|
|
|
help: "Monitor Response Time (ms)",
|
|
|
|
labelNames: this.monitorLabelNames
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (!this.monitorStatus) {
|
|
|
|
this.monitorStatus = new PrometheusClient.Gauge({
|
|
|
|
name: "monitor_status",
|
|
|
|
help: "Monitor Status (1 = UP, 0 = DOWN, 2 = PENDING, 3 = MAINTENANCE)",
|
|
|
|
labelNames: this.monitorLabelNames
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Wrapper to create a `Prometheus` instance and ensure metrics are initialized.
|
|
|
|
* @param {Monitor} monitor Monitor object to monitor
|
|
|
|
* @returns {Promise<Prometheus>} `Prometheus` instance
|
|
|
|
*/
|
|
|
|
static async createAndInitMetrics(monitor) {
|
|
|
|
await Prometheus.initMetrics();
|
|
|
|
let tags = await monitor.getTags();
|
|
|
|
return new Prometheus(monitor, tags);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a prometheus metric instance.
|
|
|
|
*
|
|
|
|
* Note: Make sure to call `Prometheus.initMetrics()` once prior creating Prometheus instances.
|
|
|
|
* @param {Monitor} monitor Monitor object to monitor
|
|
|
|
* @param {Promise<LooseObject<any>[]>} tags Tags of the monitor
|
|
|
|
*/
|
|
|
|
constructor(monitor, tags) {
|
2021-07-27 18:52:31 +02:00
|
|
|
this.monitorLabelValues = {
|
|
|
|
monitor_name: monitor.name,
|
|
|
|
monitor_type: monitor.type,
|
|
|
|
monitor_url: monitor.url,
|
|
|
|
monitor_hostname: monitor.hostname,
|
|
|
|
monitor_port: monitor.port
|
2021-10-10 10:37:53 +02:00
|
|
|
};
|
2024-08-24 18:02:57 +02:00
|
|
|
Object.values(tags)
|
|
|
|
// only label names that were known at first metric creation.
|
|
|
|
.filter(tag => Prometheus.monitorLabelNames.includes(tag.name))
|
|
|
|
.forEach(tag => {
|
|
|
|
this.monitorLabelValues[tag.name] = tag.value;
|
|
|
|
});
|
2021-07-27 18:52:31 +02:00
|
|
|
}
|
|
|
|
|
2022-04-20 20:56:40 +02:00
|
|
|
/**
|
|
|
|
* Update the metrics page
|
2023-08-11 09:46:41 +02:00
|
|
|
* @param {object} heartbeat Heartbeat details
|
|
|
|
* @param {object} tlsInfo TLS details
|
|
|
|
* @returns {void}
|
2022-04-20 20:56:40 +02:00
|
|
|
*/
|
2021-08-10 11:51:30 +02:00
|
|
|
update(heartbeat, tlsInfo) {
|
2021-08-10 14:00:59 +02:00
|
|
|
if (typeof tlsInfo !== "undefined") {
|
2021-08-10 13:14:13 +02:00
|
|
|
try {
|
2022-04-17 09:43:03 +02:00
|
|
|
let isValid;
|
|
|
|
if (tlsInfo.valid === true) {
|
2022-04-13 18:30:32 +02:00
|
|
|
isValid = 1;
|
2021-08-10 13:14:13 +02:00
|
|
|
} else {
|
2022-04-13 18:30:32 +02:00
|
|
|
isValid = 0;
|
2021-08-10 13:14:13 +02:00
|
|
|
}
|
2024-08-24 18:02:57 +02:00
|
|
|
Prometheus.monitorCertIsValid.set(this.monitorLabelValues, isValid);
|
2021-08-10 13:14:13 +02:00
|
|
|
} catch (e) {
|
2022-04-13 17:33:37 +02:00
|
|
|
log.error("prometheus", "Caught error");
|
|
|
|
log.error("prometheus", e);
|
2021-08-10 13:14:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2022-01-14 08:51:45 +02:00
|
|
|
if (tlsInfo.certInfo != null) {
|
2024-08-24 18:02:57 +02:00
|
|
|
Prometheus.monitorCertDaysRemaining.set(this.monitorLabelValues, tlsInfo.certInfo.daysRemaining);
|
2022-01-12 10:12:12 +02:00
|
|
|
}
|
2021-08-10 13:14:13 +02:00
|
|
|
} catch (e) {
|
2022-04-13 17:33:37 +02:00
|
|
|
log.error("prometheus", "Caught error");
|
|
|
|
log.error("prometheus", e);
|
2021-08-10 13:14:13 +02:00
|
|
|
}
|
|
|
|
}
|
2021-08-10 11:51:30 +02:00
|
|
|
|
2024-04-24 08:37:17 +02:00
|
|
|
if (heartbeat) {
|
|
|
|
try {
|
2024-08-24 18:02:57 +02:00
|
|
|
Prometheus.monitorStatus.set(this.monitorLabelValues, heartbeat.status);
|
2024-04-24 08:37:17 +02:00
|
|
|
} catch (e) {
|
|
|
|
log.error("prometheus", "Caught error");
|
|
|
|
log.error("prometheus", e);
|
|
|
|
}
|
2021-07-27 18:52:31 +02:00
|
|
|
|
2024-04-24 08:37:17 +02:00
|
|
|
try {
|
|
|
|
if (typeof heartbeat.ping === "number") {
|
2024-08-24 18:02:57 +02:00
|
|
|
Prometheus.monitorResponseTime.set(this.monitorLabelValues, heartbeat.ping);
|
2024-04-24 08:37:17 +02:00
|
|
|
} else {
|
|
|
|
// Is it good?
|
2024-08-24 18:02:57 +02:00
|
|
|
Prometheus.monitorResponseTime.set(this.monitorLabelValues, -1);
|
2024-04-24 08:37:17 +02:00
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
log.error("prometheus", "Caught error");
|
|
|
|
log.error("prometheus", e);
|
2021-07-27 18:52:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-11 09:46:41 +02:00
|
|
|
/**
|
|
|
|
* Remove monitor from prometheus
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
2022-01-07 05:57:24 +02:00
|
|
|
remove() {
|
|
|
|
try {
|
2024-08-24 18:02:57 +02:00
|
|
|
Prometheus.monitorCertDaysRemaining?.remove(this.monitorLabelValues);
|
|
|
|
Prometheus.monitorCertIsValid?.remove(this.monitorLabelValues);
|
|
|
|
Prometheus.monitorResponseTime?.remove(this.monitorLabelValues);
|
|
|
|
Prometheus.monitorStatus?.remove(this.monitorLabelValues);
|
2022-01-07 05:57:24 +02:00
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
|
|
|
}
|
|
|
|
}
|
2021-07-27 18:52:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
Prometheus
|
2021-10-10 10:37:53 +02:00
|
|
|
};
|