mirror of
https://github.com/louislam/uptime-kuma.git
synced 2025-01-15 17:27:36 -05:00
GH-5214: feat: Add Docker service availability monitoring
Introduces Docker service availability to monitor uptime and status of Docker services alongside existing types. Updates the database schema to support "docker_service". Enhances the user interface to allow configuration of Docker service parameters within the monitor setup. This change allows for more granular monitoring capabilities by checking the running state of services in a Docker environment. Relates to implementing extended Docker monitoring functionalities. implements #5214 Signed-off-by: Bryan Gonzalez <bryan@battleface.com>
This commit is contained in:
parent
d7ffa33950
commit
bd719d9987
@ -23,7 +23,7 @@ It is a temporary live demo, all data will be deleted after 10 minutes. Sponsore
|
|||||||
|
|
||||||
## ⭐ Features
|
## ⭐ Features
|
||||||
|
|
||||||
- Monitoring uptime for HTTP(s) / TCP / HTTP(s) Keyword / HTTP(s) Json Query / Ping / DNS Record / Push / Steam Game Server / Docker Containers
|
- Monitoring uptime for HTTP(s) / TCP / HTTP(s) Keyword / HTTP(s) Json Query / Ping / DNS Record / Push / Steam Game Server / Docker Containers / Docker Services
|
||||||
- Fancy, Reactive, Fast UI/UX
|
- Fancy, Reactive, Fast UI/UX
|
||||||
- Notifications via Telegram, Discord, Gotify, Slack, Pushover, Email (SMTP), and [90+ notification services, click here for the full list](https://github.com/louislam/uptime-kuma/tree/master/src/components/notifications)
|
- Notifications via Telegram, Discord, Gotify, Slack, Pushover, Email (SMTP), and [90+ notification services, click here for the full list](https://github.com/louislam/uptime-kuma/tree/master/src/components/notifications)
|
||||||
- 20-second intervals
|
- 20-second intervals
|
||||||
|
11
db/knex_migrations/2024-10-17-0000-add-docker-service.js
Normal file
11
db/knex_migrations/2024-10-17-0000-add-docker-service.js
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
exports.up = function (knex) {
|
||||||
|
return knex.schema.alterTable("monitor", function (table) {
|
||||||
|
table.string("docker_service", 255);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
exports.down = function (knex) {
|
||||||
|
return knex.schema.alterTable("monitor", function (table) {
|
||||||
|
table.dropColumn("docker_service");
|
||||||
|
});
|
||||||
|
};
|
@ -120,6 +120,7 @@ class Monitor extends BeanModel {
|
|||||||
dns_resolve_server: this.dns_resolve_server,
|
dns_resolve_server: this.dns_resolve_server,
|
||||||
dns_last_result: this.dns_last_result,
|
dns_last_result: this.dns_last_result,
|
||||||
docker_container: this.docker_container,
|
docker_container: this.docker_container,
|
||||||
|
docker_service: this.docker_service,
|
||||||
docker_host: this.docker_host,
|
docker_host: this.docker_host,
|
||||||
proxyId: this.proxy_id,
|
proxyId: this.proxy_id,
|
||||||
notificationIDList: preloadData.notifications.get(this.id) || {},
|
notificationIDList: preloadData.notifications.get(this.id) || {},
|
||||||
@ -754,6 +755,54 @@ class Monitor extends BeanModel {
|
|||||||
} else {
|
} else {
|
||||||
throw Error("Container State is " + res.data.State.Status);
|
throw Error("Container State is " + res.data.State.Status);
|
||||||
}
|
}
|
||||||
|
} else if (this.type === "docker-service-availability") { // TODO: add a service-health to count running vs desired (needs services api)
|
||||||
|
log.debug("monitor", `[${this.name}] Prepare Options for Axios`);
|
||||||
|
|
||||||
|
const options = {
|
||||||
|
url: `/tasks?filters={"service":{"${this.docker_service}":true}}`,
|
||||||
|
timeout: this.interval * 1000 * 0.8,
|
||||||
|
headers: {
|
||||||
|
"Accept": "*/*",
|
||||||
|
},
|
||||||
|
httpsAgent: new https.Agent({
|
||||||
|
maxCachedSessions: 0, // Use Custom agent to disable session reuse (https://github.com/nodejs/node/issues/3940)
|
||||||
|
rejectUnauthorized: !this.getIgnoreTls(),
|
||||||
|
secureOptions: crypto.constants.SSL_OP_LEGACY_SERVER_CONNECT,
|
||||||
|
}),
|
||||||
|
httpAgent: new http.Agent({
|
||||||
|
maxCachedSessions: 0,
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
|
||||||
|
const dockerHost = await R.load("docker_host", this.docker_host);
|
||||||
|
|
||||||
|
if (!dockerHost) {
|
||||||
|
throw new Error("Failed to load docker host config");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dockerHost._dockerType === "socket") {
|
||||||
|
options.socketPath = dockerHost._dockerDaemon;
|
||||||
|
} else if (dockerHost._dockerType === "tcp") {
|
||||||
|
options.baseURL = DockerHost.patchDockerURL(dockerHost._dockerDaemon);
|
||||||
|
options.httpsAgent = new https.Agent(
|
||||||
|
DockerHost.getHttpsAgentOptions(dockerHost._dockerType, options.baseURL)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
log.debug("monitor", `[${this.name}] Axios Request (get service tasks)`);
|
||||||
|
let res = await axios.request(options);
|
||||||
|
|
||||||
|
if (res.data.message) {
|
||||||
|
throw Error(res.data.message);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (res.data.filter((task) => task.Status.State === "running").length) {
|
||||||
|
bean.status = UP;
|
||||||
|
bean.msg = `Service: ${this.docker_service} is available`;
|
||||||
|
} else {
|
||||||
|
bean.status = DOWN;
|
||||||
|
bean.msg = `Service: ${this.docker_service} is Down. No replicas available`;
|
||||||
|
}
|
||||||
} else if (this.type === "sqlserver") {
|
} else if (this.type === "sqlserver") {
|
||||||
let startTime = dayjs().valueOf();
|
let startTime = dayjs().valueOf();
|
||||||
|
|
||||||
|
@ -826,6 +826,7 @@ let needSetup = false;
|
|||||||
bean.dns_resolve_server = monitor.dns_resolve_server;
|
bean.dns_resolve_server = monitor.dns_resolve_server;
|
||||||
bean.pushToken = monitor.pushToken;
|
bean.pushToken = monitor.pushToken;
|
||||||
bean.docker_container = monitor.docker_container;
|
bean.docker_container = monitor.docker_container;
|
||||||
|
bean.docker_service = monitor.docker_service;
|
||||||
bean.docker_host = monitor.docker_host;
|
bean.docker_host = monitor.docker_host;
|
||||||
bean.proxyId = Number.isInteger(monitor.proxyId) ? monitor.proxyId : null;
|
bean.proxyId = Number.isInteger(monitor.proxyId) ? monitor.proxyId : null;
|
||||||
bean.mqttUsername = monitor.mqttUsername;
|
bean.mqttUsername = monitor.mqttUsername;
|
||||||
|
@ -42,6 +42,9 @@
|
|||||||
<option value="docker">
|
<option value="docker">
|
||||||
{{ $t("Docker Container") }}
|
{{ $t("Docker Container") }}
|
||||||
</option>
|
</option>
|
||||||
|
<option value="docker-service-availability">
|
||||||
|
{{ $t("Docker Service (Simple Availability)") }}
|
||||||
|
</option>
|
||||||
|
|
||||||
<option value="real-browser">
|
<option value="real-browser">
|
||||||
HTTP(s) - Browser Engine (Chrome/Chromium) (Beta)
|
HTTP(s) - Browser Engine (Chrome/Chromium) (Beta)
|
||||||
@ -367,9 +370,16 @@
|
|||||||
<input id="docker_container" v-model="monitor.docker_container" type="text" class="form-control" required>
|
<input id="docker_container" v-model="monitor.docker_container" type="text" class="form-control" required>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!-- Docker Service Name / ID -->
|
||||||
|
<!-- For Docker Service Type -->
|
||||||
|
<div v-if="monitor.type === 'docker-service-availability'" class="my-3">
|
||||||
|
<label for="docker_service" class="form-label">{{ $t("Service Name / ID") }}</label>
|
||||||
|
<input id="docker_service" v-model="monitor.docker_service" type="text" class="form-control" required>
|
||||||
|
</div>
|
||||||
|
|
||||||
<!-- Docker Host -->
|
<!-- Docker Host -->
|
||||||
<!-- For Docker Type -->
|
<!-- For Docker Type -->
|
||||||
<div v-if="monitor.type === 'docker'" class="my-3">
|
<div v-if="['docker', 'docker-service-availability'].includes(monitor.type)" class="my-3">
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label for="docker-host" class="form-label">{{ $t("Docker Host") }}</label>
|
<label for="docker-host" class="form-label">{{ $t("Docker Host") }}</label>
|
||||||
<ActionSelect
|
<ActionSelect
|
||||||
@ -1103,6 +1113,7 @@ const monitorDefaults = {
|
|||||||
dns_resolve_type: "A",
|
dns_resolve_type: "A",
|
||||||
dns_resolve_server: "1.1.1.1",
|
dns_resolve_server: "1.1.1.1",
|
||||||
docker_container: "",
|
docker_container: "",
|
||||||
|
docker_service: "",
|
||||||
docker_host: null,
|
docker_host: null,
|
||||||
proxyId: null,
|
proxyId: null,
|
||||||
mqttUsername: "",
|
mqttUsername: "",
|
||||||
@ -1730,7 +1741,7 @@ message HealthCheckResponse {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (this.monitor.type === "docker") {
|
if ([ "docker", "docker-service-availability" ].includes(this.monitor.type)) {
|
||||||
if (this.monitor.docker_host == null) {
|
if (this.monitor.docker_host == null) {
|
||||||
toast.error(this.$t("DockerHostRequired"));
|
toast.error(this.$t("DockerHostRequired"));
|
||||||
return false;
|
return false;
|
||||||
|
Loading…
Reference in New Issue
Block a user