[BS5] add container disk and network stats
This commit is contained in:
parent
2e10cc8e79
commit
7f70b0f703
@ -378,3 +378,18 @@ table.dataTable.table-striped>tbody>tr>td>input[type="checkbox"] {
|
|||||||
.btn-check-label {
|
.btn-check-label {
|
||||||
color: #555;
|
color: #555;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.caret {
|
||||||
|
transform: rotate(0deg);
|
||||||
|
}
|
||||||
|
a[aria-expanded='true'] > .caret,
|
||||||
|
button[aria-expanded='true'] > .caret {
|
||||||
|
transform: rotate(-180deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.list-group-details {
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
.list-group-header {
|
||||||
|
background: #f7f7f7;
|
||||||
|
}
|
@ -340,3 +340,11 @@ table.dataTable.dtr-inline.collapsed>tbody>tr>td.dataTables_empty {
|
|||||||
.inputMissingAttr {
|
.inputMissingAttr {
|
||||||
border-color: #FF4136 !important;
|
border-color: #FF4136 !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.list-group-details {
|
||||||
|
background: #444444;
|
||||||
|
}
|
||||||
|
.list-group-header {
|
||||||
|
background: #333;
|
||||||
|
}
|
@ -1,6 +1,7 @@
|
|||||||
<?php
|
<?php
|
||||||
function docker($action, $service_name = null, $attr1 = null, $attr2 = null, $extra_headers = null) {
|
function docker($action, $service_name = null, $attr1 = null, $attr2 = null, $extra_headers = null) {
|
||||||
global $DOCKER_TIMEOUT;
|
global $DOCKER_TIMEOUT;
|
||||||
|
global $redis;
|
||||||
$curl = curl_init();
|
$curl = curl_init();
|
||||||
curl_setopt($curl, CURLOPT_HTTPHEADER,array('Content-Type: application/json' ));
|
curl_setopt($curl, CURLOPT_HTTPHEADER,array('Content-Type: application/json' ));
|
||||||
// We are using our mail certificates for dockerapi, the names will not match, the certs are trusted anyway
|
// We are using our mail certificates for dockerapi, the names will not match, the certs are trusted anyway
|
||||||
@ -52,6 +53,7 @@ function docker($action, $service_name = null, $attr1 = null, $attr2 = null, $ex
|
|||||||
if (strtolower($container['Config']['Labels']['com.docker.compose.project']) == strtolower(getenv('COMPOSE_PROJECT_NAME'))) {
|
if (strtolower($container['Config']['Labels']['com.docker.compose.project']) == strtolower(getenv('COMPOSE_PROJECT_NAME'))) {
|
||||||
$out[$container['Config']['Labels']['com.docker.compose.service']]['State'] = $container['State'];
|
$out[$container['Config']['Labels']['com.docker.compose.service']]['State'] = $container['State'];
|
||||||
$out[$container['Config']['Labels']['com.docker.compose.service']]['Config'] = $container['Config'];
|
$out[$container['Config']['Labels']['com.docker.compose.service']]['Config'] = $container['Config'];
|
||||||
|
$out[$container['Config']['Labels']['com.docker.compose.service']]['Id'] = trim($container['Id']);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -95,6 +97,7 @@ function docker($action, $service_name = null, $attr1 = null, $attr2 = null, $ex
|
|||||||
unset($container['Config']['Env']);
|
unset($container['Config']['Env']);
|
||||||
$out[$container['Config']['Labels']['com.docker.compose.service']]['State'] = $container['State'];
|
$out[$container['Config']['Labels']['com.docker.compose.service']]['State'] = $container['State'];
|
||||||
$out[$container['Config']['Labels']['com.docker.compose.service']]['Config'] = $container['Config'];
|
$out[$container['Config']['Labels']['com.docker.compose.service']]['Config'] = $container['Config'];
|
||||||
|
$out[$container['Config']['Labels']['com.docker.compose.service']]['Id'] = trim($container['Id']);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -104,6 +107,7 @@ function docker($action, $service_name = null, $attr1 = null, $attr2 = null, $ex
|
|||||||
unset($container['Config']['Env']);
|
unset($container['Config']['Env']);
|
||||||
$out[$decoded_response['Config']['Labels']['com.docker.compose.service']]['State'] = $decoded_response['State'];
|
$out[$decoded_response['Config']['Labels']['com.docker.compose.service']]['State'] = $decoded_response['State'];
|
||||||
$out[$decoded_response['Config']['Labels']['com.docker.compose.service']]['Config'] = $decoded_response['Config'];
|
$out[$decoded_response['Config']['Labels']['com.docker.compose.service']]['Config'] = $decoded_response['Config'];
|
||||||
|
$out[$decoded_response['Config']['Labels']['com.docker.compose.service']]['Id'] = trim($decoded_response['Id']);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -147,6 +151,29 @@ function docker($action, $service_name = null, $attr1 = null, $attr2 = null, $ex
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case 'container_stats':
|
||||||
|
if (empty($service_name)){
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$container_id = $service_name;
|
||||||
|
curl_setopt($curl, CURLOPT_URL, 'https://dockerapi:443/container/' . $container_id . '/stats/update');
|
||||||
|
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
|
||||||
|
curl_setopt($curl, CURLOPT_POST, 1);
|
||||||
|
curl_setopt($curl, CURLOPT_TIMEOUT, $DOCKER_TIMEOUT);
|
||||||
|
$response = curl_exec($curl);
|
||||||
|
if ($response === false) {
|
||||||
|
$err = curl_error($curl);
|
||||||
|
curl_close($curl);
|
||||||
|
return $err;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
curl_close($curl);
|
||||||
|
$stats = json_decode($response, true);
|
||||||
|
if (!empty($stats)) return $stats;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
break;
|
||||||
case 'host_stats':
|
case 'host_stats':
|
||||||
curl_setopt($curl, CURLOPT_URL, 'https://dockerapi:443/host/stats');
|
curl_setopt($curl, CURLOPT_URL, 'https://dockerapi:443/host/stats');
|
||||||
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
|
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
|
||||||
|
@ -37,12 +37,15 @@ $(document).ready(function() {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// set update loop container list
|
||||||
|
containersToUpdate = {}
|
||||||
// set default ChartJs Font Color
|
// set default ChartJs Font Color
|
||||||
Chart.defaults.color = '#999';
|
Chart.defaults.color = '#999';
|
||||||
// create net and disk charts
|
// create host cpu and mem charts
|
||||||
createNetAndDiskChart();
|
createHostCpuAndMemChart();
|
||||||
// check for new version
|
// check for new version
|
||||||
check_update(mailcow_info.version_tag, mailcow_info.project_url);
|
check_update(mailcow_info.version_tag, mailcow_info.project_url);
|
||||||
|
update_container_stats()
|
||||||
});
|
});
|
||||||
jQuery(function($){
|
jQuery(function($){
|
||||||
if (localStorage.getItem("current_page") === null) {
|
if (localStorage.getItem("current_page") === null) {
|
||||||
@ -1005,13 +1008,61 @@ jQuery(function($){
|
|||||||
onVisible("[id^=rspamd_history]", () => draw_rspamd_history());
|
onVisible("[id^=rspamd_history]", () => draw_rspamd_history());
|
||||||
onVisible("[id^=rspamd_donut]", () => rspamd_pie_graph());
|
onVisible("[id^=rspamd_donut]", () => rspamd_pie_graph());
|
||||||
|
|
||||||
// start polling stats if tab is active
|
|
||||||
|
|
||||||
|
// start polling host stats if tab is active
|
||||||
onVisible("[id^=tab-containers]", () => update_stats());
|
onVisible("[id^=tab-containers]", () => update_stats());
|
||||||
|
// start polling container stats if collapse is active
|
||||||
|
var containerElements = document.querySelectorAll(".container-details-collapse");
|
||||||
|
for (let i = 0; i < containerElements.length; i++){
|
||||||
|
new IntersectionObserver((entries, observer) => {
|
||||||
|
entries.forEach(entry => {
|
||||||
|
if(entry.intersectionRatio > 0) {
|
||||||
|
|
||||||
|
if (!containerElements[i].classList.contains("show")){
|
||||||
|
var container = containerElements[i].id.replace("Collapse", "");
|
||||||
|
var container_id = containerElements[i].getAttribute("data-id");
|
||||||
|
|
||||||
|
// check if chart exists or needs to be created
|
||||||
|
if (!Chart.getChart(container + "_DiskIOChart"))
|
||||||
|
createReadWriteChart(container + "_DiskIOChart", "Read", "Write");
|
||||||
|
if (!Chart.getChart(container + "_NetIOChart"))
|
||||||
|
createReadWriteChart(container + "_NetIOChart", "Recv", "Sent");
|
||||||
|
|
||||||
|
// add container to polling list
|
||||||
|
containersToUpdate[container] = {
|
||||||
|
id: container_id,
|
||||||
|
state: "idle"
|
||||||
|
}
|
||||||
|
|
||||||
|
// stop polling if collapse is closed
|
||||||
|
containerElements[i].addEventListener('hidden.bs.collapse', function () {
|
||||||
|
var diskIOCtx = Chart.getChart(container + "_DiskIOChart");
|
||||||
|
var netIOCtx = Chart.getChart(container + "_NetIOChart");
|
||||||
|
|
||||||
|
diskIOCtx.data.datasets[0].data = [];
|
||||||
|
diskIOCtx.data.datasets[1].data = [];
|
||||||
|
diskIOCtx.data.labels = [];
|
||||||
|
netIOCtx.data.datasets[0].data = [];
|
||||||
|
netIOCtx.data.datasets[1].data = [];
|
||||||
|
netIOCtx.data.labels = [];
|
||||||
|
|
||||||
|
diskIOCtx.update();
|
||||||
|
netIOCtx.update();
|
||||||
|
|
||||||
|
delete containersToUpdate[container];
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}).observe(containerElements[i]);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
// update system stats - every 5 seconds if system & container tab is active
|
// update system stats - every 5 seconds if system & container tab is active
|
||||||
function update_stats(prev_stats = null){
|
function update_stats(timeout=5){
|
||||||
if (!$('#tab-containers').hasClass('active')) {
|
if (!$('#tab-containers').hasClass('active')) {
|
||||||
// tab not active - dont fetch stats - run again in n seconds
|
// tab not active - dont fetch stats - run again in n seconds
|
||||||
return;
|
return;
|
||||||
@ -1020,62 +1071,115 @@ function update_stats(prev_stats = null){
|
|||||||
window.fetch("/api/v1/get/status/host", {method:'GET',cache:'no-cache'}).then(function(response) {
|
window.fetch("/api/v1/get/status/host", {method:'GET',cache:'no-cache'}).then(function(response) {
|
||||||
return response.json();
|
return response.json();
|
||||||
}).then(function(data) {
|
}).then(function(data) {
|
||||||
// display table data
|
console.log(data);
|
||||||
$("#host_date").text(data.system_time);
|
|
||||||
$("#host_uptime").text(formatUptime(data.uptime));
|
|
||||||
$("#host_cpu_cores").text(data.cpu.cores);
|
|
||||||
$("#host_cpu_usage").text(parseInt(data.cpu.usage).toString() + "%");
|
|
||||||
$("#host_memory_total").text((data.memory.total / (1024 ** 3)).toFixed(2).toString() + "GB");
|
|
||||||
$("#host_memory_usage").text(parseInt(data.memory.usage).toString() + "%");
|
|
||||||
|
|
||||||
|
if (data){
|
||||||
|
// display table data
|
||||||
|
$("#host_date").text(data.system_time);
|
||||||
|
$("#host_uptime").text(formatUptime(data.uptime));
|
||||||
|
$("#host_cpu_cores").text(data.cpu.cores);
|
||||||
|
$("#host_cpu_usage").text(parseInt(data.cpu.usage).toString() + "%");
|
||||||
|
$("#host_memory_total").text((data.memory.total / (1024 ** 3)).toFixed(2).toString() + "GB");
|
||||||
|
$("#host_memory_usage").text(parseInt(data.memory.usage).toString() + "%");
|
||||||
|
|
||||||
// display network and disk i/o
|
// update cpu and mem chart
|
||||||
if (prev_stats != null){
|
var cpu_chart = Chart.getChart("host_cpu_chart");
|
||||||
// get chart instances by elemId
|
var mem_chart = Chart.getChart("host_mem_chart");
|
||||||
var net_io_chart = Chart.getChart("net_io_chart");
|
|
||||||
var disk_io_chart = Chart.getChart("disk_io_chart");
|
|
||||||
|
|
||||||
|
cpu_chart.data.labels.push(data.system_time.split(" ")[1]);
|
||||||
|
if (cpu_chart.data.labels.length > 30) cpu_chart.data.labels.shift();
|
||||||
|
mem_chart.data.labels.push(data.system_time.split(" ")[1]);
|
||||||
|
if (mem_chart.data.labels.length > 30) mem_chart.data.labels.shift();
|
||||||
|
|
||||||
// calc time diff
|
cpu_chart.data.datasets[0].data.push(data.cpu.usage);
|
||||||
var time_diff = (new Date(data.system_time) - new Date(prev_stats.system_time)) / 1000;
|
if (cpu_chart.data.datasets[0].data.length > 30) cpu_chart.data.datasets[0].data.shift();
|
||||||
// push time label for x-axis
|
mem_chart.data.datasets[0].data.push(data.memory.usage);
|
||||||
net_io_chart.data.labels.push(data.system_time.split(" ")[1]);
|
if (mem_chart.data.datasets[0].data.length > 30) mem_chart.data.datasets[0].data.shift();
|
||||||
// shift data if more than 20 entires exists
|
|
||||||
if (net_io_chart.data.labels.length > 20) net_io_chart.data.labels.shift();
|
|
||||||
|
|
||||||
var diff_bytes_recv = (data.network.bytes_recv_total - prev_stats.network.bytes_recv_total) / time_diff;
|
cpu_chart.update();
|
||||||
var diff_bytes_sent = (data.network.bytes_sent_total - prev_stats.network.bytes_sent_total) / time_diff;
|
mem_chart.update();
|
||||||
net_io_chart.data.datasets[0].data.push(diff_bytes_recv);
|
|
||||||
net_io_chart.data.datasets[1].data.push(diff_bytes_sent);
|
|
||||||
// shift data if more than 20 entires exists
|
|
||||||
if (net_io_chart.data.datasets[0].data.length > 20) net_io_chart.data.datasets[0].data.shift();
|
|
||||||
if (net_io_chart.data.datasets[1].data.length > 20) net_io_chart.data.datasets[1].data.shift();
|
|
||||||
|
|
||||||
|
|
||||||
// push time label for x-axis
|
|
||||||
disk_io_chart.data.labels.push(data.system_time.split(" ")[1]);
|
|
||||||
// shift data if more than 20 entires exists
|
|
||||||
if (disk_io_chart.data.labels.length > 20) disk_io_chart.data.labels.shift();
|
|
||||||
|
|
||||||
var diff_bytes_read = (data.disk.bytes_read_total - prev_stats.disk.bytes_read_total) / time_diff;
|
|
||||||
var diff_bytes_write = (data.disk.bytes_write_total - prev_stats.disk.bytes_write_total) / time_diff;
|
|
||||||
disk_io_chart.data.datasets[0].data.push(diff_bytes_read);
|
|
||||||
disk_io_chart.data.datasets[1].data.push(diff_bytes_write);
|
|
||||||
// shift data if more than 20 entires exists
|
|
||||||
if (disk_io_chart.data.datasets[0].data.length > 20) disk_io_chart.data.datasets[0].data.shift();
|
|
||||||
if (disk_io_chart.data.datasets[1].data.length > 20) disk_io_chart.data.datasets[1].data.shift();
|
|
||||||
|
|
||||||
|
|
||||||
// update charts
|
|
||||||
net_io_chart.update();
|
|
||||||
disk_io_chart.update();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// run again in n seconds
|
// run again in n seconds
|
||||||
prev_stats = data;
|
setTimeout(update_stats, timeout * 1000);
|
||||||
setTimeout(update_stats(prev_stats), 2500);
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
// update specific container stats - every n (default 5s) seconds
|
||||||
|
function update_container_stats(timeout=5){
|
||||||
|
for (let container in containersToUpdate){
|
||||||
|
container_id = containersToUpdate[container].id;
|
||||||
|
if (containersToUpdate[container].state == "running")
|
||||||
|
continue;
|
||||||
|
containersToUpdate[container].state = "running";
|
||||||
|
|
||||||
|
|
||||||
|
window.fetch("/api/v1/get/status/container/" + container_id, {method:'GET',cache:'no-cache'}).then(function(response) {
|
||||||
|
return response.json();
|
||||||
|
}).then(function(data) {
|
||||||
|
var diskIOCtx = Chart.getChart(container + "_DiskIOChart");
|
||||||
|
var netIOCtx = Chart.getChart(container + "_NetIOChart");
|
||||||
|
|
||||||
|
console.log(container);
|
||||||
|
console.log(data);
|
||||||
|
prev_stats = null;
|
||||||
|
if (data.length >= 2)
|
||||||
|
prev_stats = data[data.length -2]
|
||||||
|
data = data[data.length -1];
|
||||||
|
|
||||||
|
if (prev_stats != null){
|
||||||
|
// calc time diff
|
||||||
|
var time_diff = (new Date(data.read) - new Date(prev_stats.read)) / 1000;
|
||||||
|
|
||||||
|
// calc disk io b/s
|
||||||
|
var prev_read_bytes = 0;
|
||||||
|
var prev_write_bytes = 0;
|
||||||
|
for (var i = 0; i < prev_stats.blkio_stats.io_service_bytes_recursive.length; i++){
|
||||||
|
if (prev_stats.blkio_stats.io_service_bytes_recursive[i].op == "read")
|
||||||
|
prev_read_bytes = prev_stats.blkio_stats.io_service_bytes_recursive[i].value;
|
||||||
|
else if (prev_stats.blkio_stats.io_service_bytes_recursive[i].op == "write")
|
||||||
|
prev_write_bytes = prev_stats.blkio_stats.io_service_bytes_recursive[i].value;
|
||||||
|
}
|
||||||
|
var read_bytes = 0;
|
||||||
|
var write_bytes = 0;
|
||||||
|
for (var i = 0; i < data.blkio_stats.io_service_bytes_recursive.length; i++){
|
||||||
|
if (data.blkio_stats.io_service_bytes_recursive[i].op == "read")
|
||||||
|
read_bytes = data.blkio_stats.io_service_bytes_recursive[i].value;
|
||||||
|
else if (data.blkio_stats.io_service_bytes_recursive[i].op == "write")
|
||||||
|
write_bytes = data.blkio_stats.io_service_bytes_recursive[i].value;
|
||||||
|
}
|
||||||
|
var diff_bytes_read = (read_bytes - prev_read_bytes) / time_diff;
|
||||||
|
var diff_bytes_write = (write_bytes - prev_write_bytes) / time_diff;
|
||||||
|
|
||||||
|
// calc net io b/s
|
||||||
|
var prev_recv_bytes = 0;
|
||||||
|
var prev_sent_bytes = 0;
|
||||||
|
for (var key in prev_stats.networks){
|
||||||
|
prev_recv_bytes += prev_stats.networks[key].rx_bytes;
|
||||||
|
prev_sent_bytes += prev_stats.networks[key].tx_bytes;
|
||||||
|
}
|
||||||
|
var recv_bytes = 0;
|
||||||
|
var sent_bytes = 0;
|
||||||
|
for (var key in data.networks){
|
||||||
|
recv_bytes += data.networks[key].rx_bytes;
|
||||||
|
sent_bytes += data.networks[key].tx_bytes;
|
||||||
|
}
|
||||||
|
var diff_bytes_recv = (recv_bytes - prev_recv_bytes) / time_diff;
|
||||||
|
var diff_bytes_sent = (sent_bytes - prev_sent_bytes) / time_diff;
|
||||||
|
|
||||||
|
addReadWriteChart(diskIOCtx, diff_bytes_read, diff_bytes_write, "");
|
||||||
|
addReadWriteChart(netIOCtx, diff_bytes_recv, diff_bytes_sent, "");
|
||||||
|
}
|
||||||
|
|
||||||
|
// run again in n seconds
|
||||||
|
containersToUpdate[container].state = "idle";
|
||||||
|
}).catch(err => {
|
||||||
|
console.log(err);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// run again in n seconds
|
||||||
|
setTimeout(update_container_stats, timeout * 1000);
|
||||||
|
}
|
||||||
// format hosts uptime seconds to readable string
|
// format hosts uptime seconds to readable string
|
||||||
function formatUptime(seconds){
|
function formatUptime(seconds){
|
||||||
seconds = Number(seconds);
|
seconds = Number(seconds);
|
||||||
@ -1103,15 +1207,14 @@ function formatBytes(bytes){
|
|||||||
// final mb to gb
|
// final mb to gb
|
||||||
return (bytes / 1024).toFixed(2).toString()+' GB/s';
|
return (bytes / 1024).toFixed(2).toString()+' GB/s';
|
||||||
}
|
}
|
||||||
// create network and disk chart
|
// create read write line chart
|
||||||
function createNetAndDiskChart(){
|
function createReadWriteChart(chart_id, read_lable, write_lable){
|
||||||
var net_io_ctx = document.getElementById("net_io_chart");
|
var ctx = document.getElementById(chart_id);
|
||||||
var disk_io_ctx = document.getElementById("disk_io_chart");
|
|
||||||
|
|
||||||
var dataNet = {
|
var dataNet = {
|
||||||
labels: [],
|
labels: [],
|
||||||
datasets: [{
|
datasets: [{
|
||||||
label: "Recieve",
|
label: read_lable,
|
||||||
backgroundColor: "rgba(41, 187, 239, 0.3)",
|
backgroundColor: "rgba(41, 187, 239, 0.3)",
|
||||||
borderColor: "rgba(41, 187, 239, 0.6)",
|
borderColor: "rgba(41, 187, 239, 0.6)",
|
||||||
pointRadius: 1,
|
pointRadius: 1,
|
||||||
@ -1121,7 +1224,7 @@ function createNetAndDiskChart(){
|
|||||||
tension: 0.2,
|
tension: 0.2,
|
||||||
data: []
|
data: []
|
||||||
}, {
|
}, {
|
||||||
label: "Sent",
|
label: write_lable,
|
||||||
backgroundColor: "rgba(239, 60, 41, 0.3)",
|
backgroundColor: "rgba(239, 60, 41, 0.3)",
|
||||||
borderColor: "rgba(239, 60, 41, 0.6)",
|
borderColor: "rgba(239, 60, 41, 0.6)",
|
||||||
pointRadius: 1,
|
pointRadius: 1,
|
||||||
@ -1156,10 +1259,36 @@ function createNetAndDiskChart(){
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
var dataDisk = {
|
return new Chart(ctx, {
|
||||||
|
type: 'line',
|
||||||
|
data: dataNet,
|
||||||
|
options: optionsNet
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// add to read write line chart
|
||||||
|
function addReadWriteChart(chart_context, read_point, write_point, time, limit = 30){
|
||||||
|
// push time label for x-axis
|
||||||
|
chart_context.data.labels.push(time);
|
||||||
|
if (chart_context.data.labels.length > limit) chart_context.data.labels.shift();
|
||||||
|
|
||||||
|
// push datapoints
|
||||||
|
chart_context.data.datasets[0].data.push(read_point);
|
||||||
|
chart_context.data.datasets[1].data.push(write_point);
|
||||||
|
// shift data if more than 20 entires exists
|
||||||
|
if (chart_context.data.datasets[0].data.length > limit) chart_context.data.datasets[0].data.shift();
|
||||||
|
if (chart_context.data.datasets[1].data.length > limit) chart_context.data.datasets[1].data.shift();
|
||||||
|
|
||||||
|
chart_context.update();
|
||||||
|
}
|
||||||
|
// create host cpu and mem chart
|
||||||
|
function createHostCpuAndMemChart(){
|
||||||
|
var cpu_ctx = document.getElementById("host_cpu_chart");
|
||||||
|
var mem_ctx = document.getElementById("host_mem_chart");
|
||||||
|
|
||||||
|
var dataCpu = {
|
||||||
labels: [],
|
labels: [],
|
||||||
datasets: [{
|
datasets: [{
|
||||||
label: "Read",
|
label: "CPU %",
|
||||||
backgroundColor: "rgba(41, 187, 239, 0.3)",
|
backgroundColor: "rgba(41, 187, 239, 0.3)",
|
||||||
borderColor: "rgba(41, 187, 239, 0.6)",
|
borderColor: "rgba(41, 187, 239, 0.6)",
|
||||||
pointRadius: 1,
|
pointRadius: 1,
|
||||||
@ -1168,19 +1297,9 @@ function createNetAndDiskChart(){
|
|||||||
fill: true,
|
fill: true,
|
||||||
tension: 0.2,
|
tension: 0.2,
|
||||||
data: []
|
data: []
|
||||||
}, {
|
|
||||||
label: "Write",
|
|
||||||
backgroundColor: "rgba(239, 60, 41, 0.3)",
|
|
||||||
borderColor: "rgba(239, 60, 41, 0.6)",
|
|
||||||
pointRadius: 1,
|
|
||||||
pointHitRadius: 6,
|
|
||||||
borderWidth: 2,
|
|
||||||
fill: true,
|
|
||||||
tension: 0.2,
|
|
||||||
data: []
|
|
||||||
}]
|
}]
|
||||||
};
|
};
|
||||||
var optionsDisk = {
|
var optionsCpu = {
|
||||||
interaction: {
|
interaction: {
|
||||||
mode: 'index'
|
mode: 'index'
|
||||||
},
|
},
|
||||||
@ -1192,7 +1311,45 @@ function createNetAndDiskChart(){
|
|||||||
},
|
},
|
||||||
ticks: {
|
ticks: {
|
||||||
callback: function(i, index, ticks) {
|
callback: function(i, index, ticks) {
|
||||||
return formatBytes(i);
|
return i.toFixed(0).toString() + "%";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
xAxis: {
|
||||||
|
grid: {
|
||||||
|
display: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var dataMem = {
|
||||||
|
labels: [],
|
||||||
|
datasets: [{
|
||||||
|
label: "MEM %",
|
||||||
|
backgroundColor: "rgba(41, 187, 239, 0.3)",
|
||||||
|
borderColor: "rgba(41, 187, 239, 0.6)",
|
||||||
|
pointRadius: 1,
|
||||||
|
pointHitRadius: 6,
|
||||||
|
borderWidth: 2,
|
||||||
|
fill: true,
|
||||||
|
tension: 0.2,
|
||||||
|
data: []
|
||||||
|
}]
|
||||||
|
};
|
||||||
|
var optionsMem = {
|
||||||
|
interaction: {
|
||||||
|
mode: 'index'
|
||||||
|
},
|
||||||
|
scales: {
|
||||||
|
yAxis: {
|
||||||
|
min: 0,
|
||||||
|
grid: {
|
||||||
|
display: false
|
||||||
|
},
|
||||||
|
ticks: {
|
||||||
|
callback: function(i, index, ticks) {
|
||||||
|
return i.toFixed(0).toString() + "%";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -1205,15 +1362,15 @@ function createNetAndDiskChart(){
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
var net_io_chart = new Chart(net_io_ctx, {
|
var net_io_chart = new Chart(cpu_ctx, {
|
||||||
type: 'line',
|
type: 'line',
|
||||||
data: dataNet,
|
data: dataCpu,
|
||||||
options: optionsNet
|
options: optionsCpu
|
||||||
});
|
});
|
||||||
var disk_io_chart = new Chart(disk_io_ctx, {
|
var disk_io_chart = new Chart(mem_ctx, {
|
||||||
type: 'line',
|
type: 'line',
|
||||||
data: dataDisk,
|
data: dataMem,
|
||||||
options: optionsDisk
|
options: optionsMem
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
// check for mailcow updates
|
// check for mailcow updates
|
||||||
|
@ -1463,6 +1463,10 @@ if (isset($_GET['query'])) {
|
|||||||
}
|
}
|
||||||
echo json_encode($temp, JSON_UNESCAPED_SLASHES);
|
echo json_encode($temp, JSON_UNESCAPED_SLASHES);
|
||||||
break;
|
break;
|
||||||
|
case "container":
|
||||||
|
$container_stats = docker('container_stats', $extra);
|
||||||
|
echo json_encode($container_stats);
|
||||||
|
break;
|
||||||
case "vmail":
|
case "vmail":
|
||||||
$exec_fields_vmail = array('cmd' => 'system', 'task' => 'df', 'dir' => '/var/vmail');
|
$exec_fields_vmail = array('cmd' => 'system', 'task' => 'df', 'dir' => '/var/vmail');
|
||||||
$vmail_df = explode(',', json_decode(docker('post', 'dovecot-mailcow', 'exec', $exec_fields_vmail), true));
|
$vmail_df = explode(',', json_decode(docker('post', 'dovecot-mailcow', 'exec', $exec_fields_vmail), true));
|
||||||
|
File diff suppressed because one or more lines are too long
Loading…
Reference in New Issue
Block a user