mirror of
https://github.com/amir20/dozzle.git
synced 2025-12-21 13:23:07 +01:00
124 lines
3.3 KiB
Vue
124 lines
3.3 KiB
Vue
<template>
|
|
<div class="flex gap-1 md:gap-4">
|
|
<StatMonitor
|
|
:data="memoryData"
|
|
label="mem"
|
|
:stat-value="formatBytes(totalStat.memoryUsage)"
|
|
:limit="formatBytes(limits.memory, { short: true, decimals: 1 })"
|
|
/>
|
|
<StatMonitor
|
|
:data="cpuData"
|
|
label="load"
|
|
:stat-value="Math.max(0, totalStat.cpu).toFixed(2) + '%'"
|
|
:limit="roundCPU(limits.cpu) + ' CPU'"
|
|
/>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts" setup>
|
|
import { Stat } from "@/models/Container";
|
|
import { Container } from "@/models/Container";
|
|
|
|
const { containers } = defineProps<{
|
|
containers: Container[];
|
|
}>();
|
|
|
|
const totalStat = ref<Stat>({ cpu: 0, memory: 0, memoryUsage: 0 });
|
|
const { history, reset } = useSimpleRefHistory(totalStat, { capacity: 300 });
|
|
const { hosts } = useHosts();
|
|
|
|
const roundCPU = (num: number) => (Number.isInteger(num) ? num.toFixed(0) : num.toFixed(1));
|
|
|
|
watch(
|
|
() => containers,
|
|
() => {
|
|
const initial: Stat[] = [];
|
|
for (let i = 1; i <= 300; i++) {
|
|
const stat = containers.reduce(
|
|
(acc, { statsHistory }) => {
|
|
const item = statsHistory.at(-i);
|
|
if (!item) {
|
|
return acc;
|
|
}
|
|
return {
|
|
cpu: acc.cpu + item.cpu,
|
|
memory: acc.memory + item.memory,
|
|
memoryUsage: acc.memoryUsage + item.memoryUsage,
|
|
};
|
|
},
|
|
{ cpu: 0, memory: 0, memoryUsage: 0 },
|
|
);
|
|
initial.push(stat);
|
|
}
|
|
reset({ initial: initial.reverse() });
|
|
},
|
|
{ immediate: true },
|
|
);
|
|
|
|
const limits = computed(() => {
|
|
const hostLimits = new Map<string, { cpu: number; memory: number }>();
|
|
|
|
for (const container of containers) {
|
|
if (!hostLimits.has(container.host)) {
|
|
hostLimits.set(container.host, {
|
|
cpu: 0,
|
|
memory: 0,
|
|
});
|
|
}
|
|
if (hostLimits.get(container.host)!.cpu < hosts.value[container.host].nCPU) {
|
|
if (container.cpuLimit == 0) {
|
|
hostLimits.get(container.host)!.cpu = hosts.value[container.host].nCPU;
|
|
} else {
|
|
hostLimits.get(container.host)!.cpu = hostLimits.get(container.host)!.cpu + container.cpuLimit;
|
|
}
|
|
}
|
|
if (hostLimits.get(container.host)!.memory < hosts.value[container.host].memTotal) {
|
|
if (container.memoryLimit == 0) {
|
|
hostLimits.get(container.host)!.memory = hosts.value[container.host].memTotal;
|
|
} else {
|
|
hostLimits.get(container.host)!.memory = hostLimits.get(container.host)!.memory + container.memoryLimit;
|
|
}
|
|
}
|
|
}
|
|
|
|
return hostLimits.values().reduce(
|
|
(acc, { cpu, memory }) => {
|
|
return {
|
|
cpu: acc.cpu + cpu,
|
|
memory: acc.memory + memory,
|
|
};
|
|
},
|
|
{ cpu: 0, memory: 0 },
|
|
);
|
|
});
|
|
|
|
useIntervalFn(() => {
|
|
totalStat.value = containers.reduce(
|
|
(acc, { stat }) => {
|
|
return {
|
|
cpu: acc.cpu + stat.cpu,
|
|
memory: acc.memory + stat.memory,
|
|
memoryUsage: acc.memoryUsage + stat.memoryUsage,
|
|
};
|
|
},
|
|
{ cpu: 0, memory: 0, memoryUsage: 0 },
|
|
);
|
|
}, 1000);
|
|
|
|
const cpuData = computed(() =>
|
|
history.value.map((stat, i) => ({
|
|
x: i,
|
|
y: Math.max(0, stat.cpu),
|
|
value: Math.max(0, stat.cpu).toFixed(2) + "%",
|
|
})),
|
|
);
|
|
|
|
const memoryData = computed(() =>
|
|
history.value.map((stat, i) => ({
|
|
x: i,
|
|
y: stat.memoryUsage,
|
|
value: formatBytes(stat.memoryUsage),
|
|
})),
|
|
);
|
|
</script>
|