mirror of
https://github.com/louislam/uptime-kuma.git
synced 2025-07-19 07:44:02 +02:00
rebucketing
This commit is contained in:
parent
75fc01c95f
commit
9402c0fa00
2 changed files with 239 additions and 43 deletions
|
@ -4,9 +4,10 @@ const { UptimeKumaServer } = require("../uptime-kuma-server");
|
||||||
const StatusPage = require("../model/status_page");
|
const StatusPage = require("../model/status_page");
|
||||||
const { allowDevAllOrigin, sendHttpError } = require("../util-server");
|
const { allowDevAllOrigin, sendHttpError } = require("../util-server");
|
||||||
const { R } = require("redbean-node");
|
const { R } = require("redbean-node");
|
||||||
const { badgeConstants } = require("../../src/util");
|
const { badgeConstants, UP, DOWN, MAINTENANCE, PENDING } = require("../../src/util");
|
||||||
const { makeBadge } = require("badge-maker");
|
const { makeBadge } = require("badge-maker");
|
||||||
const { UptimeCalculator } = require("../uptime-calculator");
|
const { UptimeCalculator } = require("../uptime-calculator");
|
||||||
|
const dayjs = require("dayjs");
|
||||||
|
|
||||||
let router = express.Router();
|
let router = express.Router();
|
||||||
|
|
||||||
|
@ -89,6 +90,8 @@ router.get("/api/status-page/heartbeat/:slug", cache("1 minutes"), async (reques
|
||||||
let heartbeatBarDays = statusPage ? (statusPage.heartbeat_bar_days || 0) : 0;
|
let heartbeatBarDays = statusPage ? (statusPage.heartbeat_bar_days || 0) : 0;
|
||||||
|
|
||||||
for (let monitorID of monitorIDList) {
|
for (let monitorID of monitorIDList) {
|
||||||
|
const uptimeCalculator = await UptimeCalculator.getUptimeCalculator(monitorID);
|
||||||
|
|
||||||
if (heartbeatBarDays === 0) {
|
if (heartbeatBarDays === 0) {
|
||||||
// Auto mode - use original LIMIT 100 logic
|
// Auto mode - use original LIMIT 100 logic
|
||||||
let list = await R.getAll(`
|
let list = await R.getAll(`
|
||||||
|
@ -103,26 +106,10 @@ router.get("/api/status-page/heartbeat/:slug", cache("1 minutes"), async (reques
|
||||||
list = R.convertToBeans("heartbeat", list);
|
list = R.convertToBeans("heartbeat", list);
|
||||||
heartbeatList[monitorID] = list.reverse().map(row => row.toPublicJSON());
|
heartbeatList[monitorID] = list.reverse().map(row => row.toPublicJSON());
|
||||||
} else {
|
} else {
|
||||||
// For configured day ranges, always use raw heartbeat data for client-side aggregation
|
// For configured day ranges, use aggregated data from UptimeCalculator
|
||||||
// This ensures consistent behavior between edit mode and published mode
|
heartbeatList[monitorID] = await getAggregatedHeartbeats(uptimeCalculator, heartbeatBarDays);
|
||||||
const date = new Date();
|
|
||||||
date.setDate(date.getDate() - heartbeatBarDays);
|
|
||||||
const dateFrom = date.toISOString().slice(0, 19).replace("T", " ");
|
|
||||||
|
|
||||||
let list = await R.getAll(`
|
|
||||||
SELECT * FROM heartbeat
|
|
||||||
WHERE monitor_id = ? AND time >= ?
|
|
||||||
ORDER BY time DESC
|
|
||||||
`, [
|
|
||||||
monitorID,
|
|
||||||
dateFrom
|
|
||||||
]);
|
|
||||||
|
|
||||||
list = R.convertToBeans("heartbeat", list);
|
|
||||||
heartbeatList[monitorID] = list.reverse().map(row => row.toPublicJSON());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const uptimeCalculator = await UptimeCalculator.getUptimeCalculator(monitorID);
|
|
||||||
uptimeList[`${monitorID}_24`] = uptimeCalculator.get24Hour().uptime;
|
uptimeList[`${monitorID}_24`] = uptimeCalculator.get24Hour().uptime;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -269,4 +256,148 @@ router.get("/api/status-page/:slug/badge", cache("5 minutes"), async (request, r
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get aggregated heartbeats for status page display
|
||||||
|
* @param {UptimeCalculator} uptimeCalculator The uptime calculator instance
|
||||||
|
* @param {number} days Number of days to show
|
||||||
|
* @returns {Promise<Array>} Array of aggregated heartbeat data
|
||||||
|
*/
|
||||||
|
async function getAggregatedHeartbeats(uptimeCalculator, days) {
|
||||||
|
const targetBuckets = 100; // Always show ~100 buckets for consistent display
|
||||||
|
const now = dayjs.utc();
|
||||||
|
const result = [];
|
||||||
|
|
||||||
|
// Determine data granularity based on days
|
||||||
|
let dataPoints;
|
||||||
|
let granularity;
|
||||||
|
let bucketSizeMinutes;
|
||||||
|
|
||||||
|
if (days <= 1) {
|
||||||
|
// For 1 day or less, use minutely data
|
||||||
|
granularity = "minute";
|
||||||
|
dataPoints = uptimeCalculator.getDataArray(days * 24 * 60, granularity);
|
||||||
|
bucketSizeMinutes = Math.max(1, Math.floor((days * 24 * 60) / targetBuckets));
|
||||||
|
} else if (days <= 30) {
|
||||||
|
// For 2-30 days, use hourly data
|
||||||
|
granularity = "hour";
|
||||||
|
dataPoints = uptimeCalculator.getDataArray(days * 24, granularity);
|
||||||
|
bucketSizeMinutes = Math.max(60, Math.floor((days * 24 * 60) / targetBuckets));
|
||||||
|
} else {
|
||||||
|
// For 31+ days, use daily data
|
||||||
|
granularity = "day";
|
||||||
|
dataPoints = uptimeCalculator.getDataArray(days, granularity);
|
||||||
|
bucketSizeMinutes = Math.max(1440, Math.floor((days * 24 * 60) / targetBuckets));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create time buckets
|
||||||
|
const startTime = now.subtract(days, "day").startOf("minute");
|
||||||
|
const endTime = now;
|
||||||
|
const buckets = [];
|
||||||
|
|
||||||
|
for (let i = 0; i < targetBuckets; i++) {
|
||||||
|
const bucketStart = startTime.add(i * bucketSizeMinutes, "minute");
|
||||||
|
let bucketEnd = bucketStart.add(bucketSizeMinutes, "minute");
|
||||||
|
|
||||||
|
// Don't create buckets that start after current time
|
||||||
|
if (bucketStart.isAfter(endTime)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure bucket doesn't extend beyond current time
|
||||||
|
if (bucketEnd.isAfter(endTime)) {
|
||||||
|
bucketEnd = endTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
buckets.push({
|
||||||
|
start: bucketStart.unix(),
|
||||||
|
end: bucketEnd.unix(),
|
||||||
|
up: 0,
|
||||||
|
down: 0,
|
||||||
|
maintenance: 0,
|
||||||
|
pending: 0,
|
||||||
|
hasData: false
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Aggregate data points into buckets
|
||||||
|
for (const dataPoint of dataPoints) {
|
||||||
|
if (!dataPoint || !dataPoint.timestamp) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find the appropriate bucket for this data point
|
||||||
|
const bucket = buckets.find(b =>
|
||||||
|
dataPoint.timestamp >= b.start && dataPoint.timestamp < b.end
|
||||||
|
);
|
||||||
|
|
||||||
|
if (bucket) {
|
||||||
|
bucket.up += dataPoint.up || 0;
|
||||||
|
bucket.down += dataPoint.down || 0;
|
||||||
|
bucket.maintenance += dataPoint.maintenance || 0;
|
||||||
|
bucket.pending += dataPoint.pending || 0;
|
||||||
|
bucket.hasData = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Convert buckets to heartbeat format
|
||||||
|
for (const bucket of buckets) {
|
||||||
|
let status = null; // No data
|
||||||
|
|
||||||
|
if (bucket.hasData) {
|
||||||
|
// Determine status based on priority: DOWN > MAINTENANCE > PENDING > UP
|
||||||
|
if (bucket.down > 0) {
|
||||||
|
status = DOWN;
|
||||||
|
} else if (bucket.maintenance > 0) {
|
||||||
|
status = MAINTENANCE;
|
||||||
|
} else if (bucket.pending > 0) {
|
||||||
|
status = PENDING;
|
||||||
|
} else if (bucket.up > 0) {
|
||||||
|
status = UP;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
result.push({
|
||||||
|
status: status,
|
||||||
|
time: dayjs.unix(bucket.end).toISOString(),
|
||||||
|
msg: "",
|
||||||
|
ping: null,
|
||||||
|
// Include aggregation info for client-side display
|
||||||
|
_aggregated: true,
|
||||||
|
_startTime: dayjs.unix(bucket.start).toISOString(),
|
||||||
|
_endTime: dayjs.unix(bucket.end).toISOString(),
|
||||||
|
_counts: {
|
||||||
|
up: bucket.up,
|
||||||
|
down: bucket.down,
|
||||||
|
maintenance: bucket.maintenance,
|
||||||
|
pending: bucket.pending
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure we always return targetBuckets number of items by padding at the start
|
||||||
|
while (result.length < targetBuckets) {
|
||||||
|
const firstStartTime = result.length > 0 ? dayjs(result[0]._startTime || result[0].time) : now.subtract(days, "day");
|
||||||
|
const paddedStart = firstStartTime.subtract(bucketSizeMinutes, "minute");
|
||||||
|
const paddedEnd = firstStartTime;
|
||||||
|
|
||||||
|
result.unshift({
|
||||||
|
status: null,
|
||||||
|
time: paddedEnd.toISOString(),
|
||||||
|
msg: "",
|
||||||
|
ping: null,
|
||||||
|
_aggregated: true,
|
||||||
|
_startTime: paddedStart.toISOString(),
|
||||||
|
_endTime: paddedEnd.toISOString(),
|
||||||
|
_counts: {
|
||||||
|
up: 0,
|
||||||
|
down: 0,
|
||||||
|
maintenance: 0,
|
||||||
|
pending: 0
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
module.exports = router;
|
module.exports = router;
|
||||||
|
|
|
@ -112,10 +112,20 @@ export default {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
// If heartbeat days is configured (not auto), always use client-side aggregation
|
// If heartbeat days is configured (not auto), check if data is already aggregated
|
||||||
// This ensures consistent behavior between edit mode and published mode
|
|
||||||
if (this.normalizedHeartbeatBarDays > 0) {
|
if (this.normalizedHeartbeatBarDays > 0) {
|
||||||
return this.aggregatedBeatList;
|
// Check if the data is already aggregated from the server
|
||||||
|
if (this.beatList.length > 0 && this.beatList[0]._aggregated) {
|
||||||
|
// Data is already aggregated from server, use it directly
|
||||||
|
// But still limit to maxBeat for display
|
||||||
|
if (this.beatList.length > this.maxBeat) {
|
||||||
|
return this.beatList.slice(this.beatList.length - this.maxBeat);
|
||||||
|
}
|
||||||
|
return this.beatList;
|
||||||
|
} else {
|
||||||
|
// Fallback to client-side aggregation for edit mode
|
||||||
|
return this.aggregatedBeatList;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Original logic for auto mode (heartbeatBarDays = 0)
|
// Original logic for auto mode (heartbeatBarDays = 0)
|
||||||
|
@ -143,28 +153,38 @@ export default {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Always do client-side aggregation using dynamic maxBeat for proper screen sizing
|
// Always do client-side aggregation using fixed time buckets
|
||||||
const now = dayjs();
|
const now = dayjs();
|
||||||
const buckets = [];
|
const buckets = [];
|
||||||
|
|
||||||
// Calculate total hours from days
|
// Use same logic as server-side: 100 buckets
|
||||||
const totalHours = this.normalizedHeartbeatBarDays * 24;
|
const targetBuckets = 100;
|
||||||
|
const days = this.normalizedHeartbeatBarDays;
|
||||||
// Use dynamic maxBeat calculated from screen size
|
const bucketSizeMinutes = Math.max(1, Math.floor((days * 24 * 60) / targetBuckets));
|
||||||
const totalBuckets = this.maxBeat > 0 ? this.maxBeat : 50;
|
|
||||||
const bucketSize = totalHours / totalBuckets;
|
|
||||||
|
|
||||||
// Create time buckets from oldest to newest
|
// Create time buckets from oldest to newest
|
||||||
const startTime = now.subtract(totalHours, "hours");
|
const startTime = now.subtract(days, "day").startOf("minute");
|
||||||
for (let i = 0; i < totalBuckets; i++) {
|
const endTime = now;
|
||||||
let bucketStart = startTime.add(i * bucketSize, "hours");
|
|
||||||
let bucketEnd = bucketStart.add(bucketSize, "hours");
|
for (let i = 0; i < targetBuckets; i++) {
|
||||||
|
let bucketStart = startTime.add(i * bucketSizeMinutes, "minute");
|
||||||
|
let bucketEnd = bucketStart.add(bucketSizeMinutes, "minute");
|
||||||
|
|
||||||
|
// Don't create buckets that start after current time
|
||||||
|
if (bucketStart.isAfter(endTime)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure bucket doesn't extend beyond current time
|
||||||
|
if (bucketEnd.isAfter(endTime)) {
|
||||||
|
bucketEnd = endTime;
|
||||||
|
}
|
||||||
|
|
||||||
buckets.push({
|
buckets.push({
|
||||||
start: bucketStart,
|
start: bucketStart,
|
||||||
end: bucketEnd,
|
end: bucketEnd,
|
||||||
beats: [],
|
beats: [],
|
||||||
status: 1, // default to up
|
status: null, // default to no data
|
||||||
time: bucketEnd.toISOString()
|
time: bucketEnd.toISOString()
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -173,8 +193,7 @@ export default {
|
||||||
this.beatList.forEach(beat => {
|
this.beatList.forEach(beat => {
|
||||||
const beatTime = dayjs.utc(beat.time).local();
|
const beatTime = dayjs.utc(beat.time).local();
|
||||||
const bucket = buckets.find(b =>
|
const bucket = buckets.find(b =>
|
||||||
(beatTime.isAfter(b.start) || beatTime.isSame(b.start)) &&
|
beatTime.unix() >= b.start.unix() && beatTime.unix() < b.end.unix()
|
||||||
(beatTime.isBefore(b.end) || beatTime.isSame(b.end))
|
|
||||||
);
|
);
|
||||||
if (bucket) {
|
if (bucket) {
|
||||||
bucket.beats.push(beat);
|
bucket.beats.push(beat);
|
||||||
|
@ -186,26 +205,46 @@ export default {
|
||||||
if (bucket.beats.length === 0) {
|
if (bucket.beats.length === 0) {
|
||||||
bucket.status = null; // no data - will be rendered as empty/grey
|
bucket.status = null; // no data - will be rendered as empty/grey
|
||||||
} else {
|
} else {
|
||||||
// Priority: DOWN (0) > MAINTENANCE (3) > UP (1)
|
// Priority: DOWN (0) > MAINTENANCE (3) > PENDING (2) > UP (1)
|
||||||
const hasDown = bucket.beats.some(b => b.status === 0);
|
const hasDown = bucket.beats.some(b => b.status === 0);
|
||||||
const hasMaintenance = bucket.beats.some(b => b.status === 3);
|
const hasMaintenance = bucket.beats.some(b => b.status === 3);
|
||||||
|
const hasPending = bucket.beats.some(b => b.status === 2);
|
||||||
|
|
||||||
if (hasDown) {
|
if (hasDown) {
|
||||||
bucket.status = 0;
|
bucket.status = 0;
|
||||||
} else if (hasMaintenance) {
|
} else if (hasMaintenance) {
|
||||||
bucket.status = 3;
|
bucket.status = 3;
|
||||||
|
} else if (hasPending) {
|
||||||
|
bucket.status = 2;
|
||||||
} else {
|
} else {
|
||||||
bucket.status = 1;
|
bucket.status = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Use the latest beat time in the bucket
|
// Use the bucket end time for consistency
|
||||||
const latestBeat = bucket.beats.reduce((latest, beat) =>
|
bucket.time = bucket.end.toISOString();
|
||||||
dayjs(beat.time).isAfter(dayjs(latest.time)) ? beat : latest
|
|
||||||
);
|
|
||||||
bucket.time = latestBeat.time;
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Ensure we always return targetBuckets number of items by padding at the start
|
||||||
|
while (buckets.length < targetBuckets) {
|
||||||
|
const firstStart = buckets.length > 0 ? buckets[0].start : now.subtract(days, "day");
|
||||||
|
const paddedStart = firstStart.subtract(bucketSizeMinutes, "minute");
|
||||||
|
const paddedEnd = firstStart;
|
||||||
|
|
||||||
|
buckets.unshift({
|
||||||
|
start: paddedStart,
|
||||||
|
end: paddedEnd,
|
||||||
|
beats: [],
|
||||||
|
status: null,
|
||||||
|
time: paddedEnd.toISOString()
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Limit to maxBeat for display
|
||||||
|
if (buckets.length > this.maxBeat) {
|
||||||
|
return buckets.slice(buckets.length - this.maxBeat);
|
||||||
|
}
|
||||||
|
|
||||||
return buckets;
|
return buckets;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -375,7 +414,7 @@ export default {
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
// For aggregated beats, show time range and status
|
// For client-side aggregated beats (edit mode)
|
||||||
if (beat.beats !== undefined) {
|
if (beat.beats !== undefined) {
|
||||||
const start = this.$root.datetime(beat.start);
|
const start = this.$root.datetime(beat.start);
|
||||||
const end = this.$root.datetime(beat.end);
|
const end = this.$root.datetime(beat.end);
|
||||||
|
@ -383,6 +422,32 @@ export default {
|
||||||
return `${start} - ${end}: ${statusText} (${beat.beats.length} checks)`;
|
return `${start} - ${end}: ${statusText} (${beat.beats.length} checks)`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// For server-side aggregated beats
|
||||||
|
if (beat._aggregated && beat._counts) {
|
||||||
|
const counts = beat._counts;
|
||||||
|
const total = counts.up + counts.down + counts.maintenance + counts.pending;
|
||||||
|
|
||||||
|
// Use start time for display if available
|
||||||
|
const displayTime = beat._startTime ? beat._startTime : beat.time;
|
||||||
|
|
||||||
|
if (total === 0) {
|
||||||
|
return `${this.$root.datetime(displayTime)}: No Data`;
|
||||||
|
}
|
||||||
|
|
||||||
|
let statusText = "";
|
||||||
|
if (counts.down > 0) {
|
||||||
|
statusText = "Down";
|
||||||
|
} else if (counts.maintenance > 0) {
|
||||||
|
statusText = "Maintenance";
|
||||||
|
} else if (counts.pending > 0) {
|
||||||
|
statusText = "Pending";
|
||||||
|
} else if (counts.up > 0) {
|
||||||
|
statusText = "Up";
|
||||||
|
}
|
||||||
|
|
||||||
|
return `${this.$root.datetime(displayTime)}: ${statusText} (${total} checks)`;
|
||||||
|
}
|
||||||
|
|
||||||
// For individual beats, show timestamp
|
// For individual beats, show timestamp
|
||||||
return `${this.$root.datetime(beat.time)}${beat.msg ? ` - ${beat.msg}` : ""}`;
|
return `${this.$root.datetime(beat.time)}${beat.msg ? ` - ${beat.msg}` : ""}`;
|
||||||
},
|
},
|
||||||
|
|
Loading…
Add table
Reference in a new issue