From 533bc1505b78297a4a9be86a95aafbdecc8db7e3 Mon Sep 17 00:00:00 2001 From: Louis Lam Date: Thu, 9 Mar 2023 22:03:23 +0800 Subject: [PATCH] Prevent generating duplicated timeslots --- server/model/maintenance_timeslot.js | 29 +++++++++++++++++++++++++--- server/uptime-kuma-server.js | 1 + 2 files changed, 27 insertions(+), 3 deletions(-) diff --git a/server/model/maintenance_timeslot.js b/server/model/maintenance_timeslot.js index 3e7076d8a..dad719c74 100644 --- a/server/model/maintenance_timeslot.js +++ b/server/model/maintenance_timeslot.js @@ -58,7 +58,14 @@ class MaintenanceTimeslot extends BeanModel { bean.start_date = maintenance.start_date; bean.end_date = maintenance.end_date; bean.generated_next = true; - return await R.store(bean); + + if (!await this.isDuplicateTimeslot(bean)) { + await R.store(bean); + return bean; + } else { + log.debug("maintenance", "Duplicate timeslot, skip"); + return null; + } } else if (maintenance.strategy === "recurring-interval") { // Prevent dead loop, in case interval_day is not set @@ -144,6 +151,15 @@ class MaintenanceTimeslot extends BeanModel { } } + static async isDuplicateTimeslot(timeslot) { + let bean = await R.findOne("maintenance_timeslot", "maintenance_id = ? AND start_date = ? AND end_date = ?", [ + timeslot.maintenance_id, + timeslot.start_date, + timeslot.end_date + ]); + return bean !== null; + } + /** * Generate a next timeslot for all recurring types * @param maintenance @@ -161,7 +177,7 @@ class MaintenanceTimeslot extends BeanModel { // Keep generating from the first possible date, until it is ok while (true) { - log.debug("timeslot", "startDateTime: " + startDateTime.format()); + //log.debug("timeslot", "startDateTime: " + startDateTime.format()); // Handling out of effective date range if (startDateTime.diff(dayjs.utc(maintenance.end_date)) > 0) { @@ -193,7 +209,14 @@ class MaintenanceTimeslot extends BeanModel { bean.start_date = localToUTC(startDateTime); bean.end_date = localToUTC(endDateTime); bean.generated_next = false; - return await R.store(bean); + + if (!await this.isDuplicateTimeslot(bean)) { + await R.store(bean); + return bean; + } else { + log.debug("maintenance", "Duplicate timeslot, skip"); + return null; + } } } diff --git a/server/uptime-kuma-server.js b/server/uptime-kuma-server.js index a9c2289e7..3dd7ba936 100644 --- a/server/uptime-kuma-server.js +++ b/server/uptime-kuma-server.js @@ -271,6 +271,7 @@ class UptimeKumaServer { /** Load the timeslots for maintenance */ async generateMaintenanceTimeslots() { + log.debug("maintenance", "Routine: Generating Maintenance Timeslots"); // Prevent #2776 // Remove duplicate maintenance_timeslot with same start_date, end_date and maintenance_id