mirror of
https://github.com/wah-su/wah-su.github.io.git
synced 2025-04-06 00:04:40 +00:00
feat: add service status widget.
This commit is contained in:
parent
adaa5a8c9b
commit
2d0a1adf42
4 changed files with 102 additions and 46 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -1 +1,2 @@
|
||||||
node_modules
|
node_modules
|
||||||
|
TODO.md
|
||||||
|
|
13
index.html
13
index.html
|
@ -12,6 +12,16 @@
|
||||||
<header class="shadow-lg pt-4 pb-6 sticky top-0 left-0">
|
<header class="shadow-lg pt-4 pb-6 sticky top-0 left-0">
|
||||||
<div class="flex flex-row gap-2 items-center container max-w-[1440px] justify-between">
|
<div class="flex flex-row gap-2 items-center container max-w-[1440px] justify-between">
|
||||||
<div><img src="/public/images/logo-light.svg" alt="wah.su" /></div>
|
<div><img src="/public/images/logo-light.svg" alt="wah.su" /></div>
|
||||||
|
<div>
|
||||||
|
<a href="https://status.wah.su" class="flex flex-row items-center gap-2" target="_blank">
|
||||||
|
<div class="bg-green-500 w-4 h-4 rounded-full hidden" id="status-up-icon"></div>
|
||||||
|
<div class="bg-yellow-400 w-4 h-4 rounded-full hidden" id="status-degraded-icon"></div>
|
||||||
|
<div class="bg-red-600 w-4 h-4 rounded-full hidden" id="status-down-icon"></div>
|
||||||
|
<p class="hidden" id="status-up-text">All Systems Operational.</p>
|
||||||
|
<p class="hidden" id="status-degraded-text">Degraded Services.</p>
|
||||||
|
<p class="hidden" id="status-down-text">All Systems Down.</p>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
<div class="flex flex-row items-center justify-center">
|
<div class="flex flex-row items-center justify-center">
|
||||||
<button type="button"
|
<button type="button"
|
||||||
class="text-white bg-blue-700 hover:bg-blue-800 focus:ring-4 focus:outline-none focus:ring-blue-300 font-medium rounded-lg text-sm px-5 py-2.5 text-center inline-flex items-center dark:bg-blue-600 dark:hover:bg-blue-700 dark:focus:ring-blue-800">
|
class="text-white bg-blue-700 hover:bg-blue-800 focus:ring-4 focus:outline-none focus:ring-blue-300 font-medium rounded-lg text-sm px-5 py-2.5 text-center inline-flex items-center dark:bg-blue-600 dark:hover:bg-blue-700 dark:focus:ring-blue-800">
|
||||||
|
@ -34,7 +44,7 @@
|
||||||
to the right place! wah.su is all about providing a safe and stable
|
to the right place! wah.su is all about providing a safe and stable
|
||||||
environment for your projects.
|
environment for your projects.
|
||||||
|
|
||||||
Run by yours truly, @radiquum, we believe in quality over quantity, which is why access is currently invite-only.
|
Run by yours truly, <a href="https://bento.me/radiquum" class="font-medium text-blue-600 dark:text-blue-500 hover:underline" target="_blank">@radiquum</a>, we believe in quality over quantity, which is why access is currently invite-only.
|
||||||
Interested in joining the pack? Reach out to me to sniff out an invitation.
|
Interested in joining the pack? Reach out to me to sniff out an invitation.
|
||||||
</p>
|
</p>
|
||||||
<!--noformat-->
|
<!--noformat-->
|
||||||
|
@ -48,6 +58,7 @@
|
||||||
|
|
||||||
<script src="/public/js/flowbite.min.js"></script>
|
<script src="/public/js/flowbite.min.js"></script>
|
||||||
<script src="/public/js/cards.js"></script>
|
<script src="/public/js/cards.js"></script>
|
||||||
|
<script src="/public/js/status.js"></script>
|
||||||
</body>
|
</body>
|
||||||
|
|
||||||
</html>
|
</html>
|
66
public/js/status.js
Normal file
66
public/js/status.js
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
const serviceUp = {
|
||||||
|
icon: document.getElementById("status-up-icon"),
|
||||||
|
text: document.getElementById("status-up-text"),
|
||||||
|
};
|
||||||
|
const serviceDegraded = {
|
||||||
|
icon: document.getElementById("status-degraded-icon"),
|
||||||
|
text: document.getElementById("status-degraded-text"),
|
||||||
|
};
|
||||||
|
const serviceDown = {
|
||||||
|
icon: document.getElementById("status-down-icon"),
|
||||||
|
text: document.getElementById("status-down-text"),
|
||||||
|
};
|
||||||
|
|
||||||
|
async function getServicesHealth() {
|
||||||
|
const serviceStatus = await fetch(
|
||||||
|
"https://status.wah.su/api/status-page/heartbeat/services"
|
||||||
|
);
|
||||||
|
|
||||||
|
const services = await serviceStatus.json();
|
||||||
|
const heartbeatDict = services.heartbeatList;
|
||||||
|
let lastHeartbeats = [];
|
||||||
|
|
||||||
|
for (const [key, value] of Object.entries(heartbeatDict)) {
|
||||||
|
lastHeartbeats.push(
|
||||||
|
heartbeatDict[key][heartbeatDict[key].length - 1].status
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
let status = "up";
|
||||||
|
const count = lastHeartbeats.reduce((partialSum, a) => partialSum + a, 0);
|
||||||
|
|
||||||
|
if (count === lastHeartbeats.length) {
|
||||||
|
status = "up";
|
||||||
|
} else if (count === 0) {
|
||||||
|
status = "down";
|
||||||
|
} else {
|
||||||
|
status = "degraded";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status === "up") {
|
||||||
|
serviceUp.icon.classList.remove("hidden");
|
||||||
|
serviceUp.text.classList.remove("hidden");
|
||||||
|
serviceDegraded.icon.classList.add("hidden");
|
||||||
|
serviceDegraded.text.classList.add("hidden");
|
||||||
|
serviceDown.icon.classList.add("hidden");
|
||||||
|
serviceDown.text.classList.add("hidden");
|
||||||
|
} else if (status === "degraded") {
|
||||||
|
serviceUp.icon.classList.add("hidden");
|
||||||
|
serviceUp.text.classList.add("hidden");
|
||||||
|
serviceDegraded.icon.classList.remove("hidden");
|
||||||
|
serviceDegraded.text.classList.remove("hidden");
|
||||||
|
serviceDown.icon.classList.add("hidden");
|
||||||
|
serviceDown.text.classList.add("hidden");
|
||||||
|
} else if (status === "down") {
|
||||||
|
serviceUp.icon.classList.add("hidden");
|
||||||
|
serviceUp.text.classList.add("hidden");
|
||||||
|
serviceDegraded.icon.classList.add("hidden");
|
||||||
|
serviceDegraded.text.classList.add("hidden");
|
||||||
|
serviceDown.icon.classList.remove("hidden");
|
||||||
|
serviceDown.text.classList.remove("hidden");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getServicesHealth();
|
||||||
|
|
||||||
|
setInterval(getServicesHealth, 600000);
|
|
@ -1162,10 +1162,6 @@ input:checked + .toggle-bg {
|
||||||
margin-top: 3.5rem;
|
margin-top: 3.5rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.mt-16 {
|
|
||||||
margin-top: 4rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mt-2 {
|
.mt-2 {
|
||||||
margin-top: 0.5rem;
|
margin-top: 0.5rem;
|
||||||
}
|
}
|
||||||
|
@ -1174,34 +1170,6 @@ input:checked + .toggle-bg {
|
||||||
margin-top: 1.5rem;
|
margin-top: 1.5rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.mt-12 {
|
|
||||||
margin-top: 3rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mt-10 {
|
|
||||||
margin-top: 2.5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mt-8 {
|
|
||||||
margin-top: 2rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mt-20 {
|
|
||||||
margin-top: 5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mt-24 {
|
|
||||||
margin-top: 6rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mt-32 {
|
|
||||||
margin-top: 8rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mt-28 {
|
|
||||||
margin-top: 7rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.block {
|
.block {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
@ -1422,6 +1390,10 @@ input:checked + .toggle-bg {
|
||||||
border-radius: 0.5rem;
|
border-radius: 0.5rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.rounded-full {
|
||||||
|
border-radius: 9999px;
|
||||||
|
}
|
||||||
|
|
||||||
.rounded-e-lg {
|
.rounded-e-lg {
|
||||||
border-start-end-radius: 0.5rem;
|
border-start-end-radius: 0.5rem;
|
||||||
border-end-end-radius: 0.5rem;
|
border-end-end-radius: 0.5rem;
|
||||||
|
@ -1513,6 +1485,21 @@ input:checked + .toggle-bg {
|
||||||
background-color: rgb(255 255 255 / 0.5);
|
background-color: rgb(255 255 255 / 0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.bg-green-500 {
|
||||||
|
--tw-bg-opacity: 1;
|
||||||
|
background-color: rgb(14 159 110 / var(--tw-bg-opacity));
|
||||||
|
}
|
||||||
|
|
||||||
|
.bg-red-600 {
|
||||||
|
--tw-bg-opacity: 1;
|
||||||
|
background-color: rgb(224 36 36 / var(--tw-bg-opacity));
|
||||||
|
}
|
||||||
|
|
||||||
|
.bg-yellow-400 {
|
||||||
|
--tw-bg-opacity: 1;
|
||||||
|
background-color: rgb(227 160 8 / var(--tw-bg-opacity));
|
||||||
|
}
|
||||||
|
|
||||||
.object-cover {
|
.object-cover {
|
||||||
-o-object-fit: cover;
|
-o-object-fit: cover;
|
||||||
object-fit: cover;
|
object-fit: cover;
|
||||||
|
@ -1563,11 +1550,6 @@ input:checked + .toggle-bg {
|
||||||
padding-bottom: 0.75rem;
|
padding-bottom: 0.75rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.py-4 {
|
|
||||||
padding-top: 1rem;
|
|
||||||
padding-bottom: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.pb-6 {
|
.pb-6 {
|
||||||
padding-bottom: 1.5rem;
|
padding-bottom: 1.5rem;
|
||||||
}
|
}
|
||||||
|
@ -1576,14 +1558,6 @@ input:checked + .toggle-bg {
|
||||||
padding-top: 0.5rem;
|
padding-top: 0.5rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pt-8 {
|
|
||||||
padding-top: 2rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.pt-6 {
|
|
||||||
padding-top: 1.5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.pt-4 {
|
.pt-4 {
|
||||||
padding-top: 1rem;
|
padding-top: 1rem;
|
||||||
}
|
}
|
||||||
|
@ -1776,6 +1750,10 @@ input:checked + .toggle-bg {
|
||||||
color: rgb(17 24 39 / var(--tw-text-opacity));
|
color: rgb(17 24 39 / var(--tw-text-opacity));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.hover\:underline:hover {
|
||||||
|
text-decoration-line: underline;
|
||||||
|
}
|
||||||
|
|
||||||
.focus\:z-10:focus {
|
.focus\:z-10:focus {
|
||||||
z-index: 10;
|
z-index: 10;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue