Compare commits
No commits in common. "d9549454e1f971a3cac53ae62f61cd99f635f71a" and "1dc39decf749bfa961565194f258f806c3a65121" have entirely different histories.
d9549454e1
...
1dc39decf7
141
app.js
141
app.js
@ -20,10 +20,9 @@ const connections = {};
|
|||||||
window.openTerminals = {};
|
window.openTerminals = {};
|
||||||
let activePeer = null;
|
let activePeer = null;
|
||||||
window.activePeer = null; // Expose to other modules
|
window.activePeer = null; // Expose to other modules
|
||||||
|
|
||||||
hideStatusIndicator();
|
hideStatusIndicator();
|
||||||
let statsInterval = null;
|
let statsInterval = null; // Global variable to hold the interval
|
||||||
let lastStatsUpdate = Date.now();
|
|
||||||
function stopStatsInterval() {
|
function stopStatsInterval() {
|
||||||
if (statsInterval) {
|
if (statsInterval) {
|
||||||
clearInterval(statsInterval);
|
clearInterval(statsInterval);
|
||||||
@ -32,41 +31,21 @@ function stopStatsInterval() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function startStatsInterval() {
|
function startStatsInterval() {
|
||||||
|
// Clear any existing interval
|
||||||
if (statsInterval) {
|
if (statsInterval) {
|
||||||
clearInterval(statsInterval);
|
clearInterval(statsInterval);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Start a new interval to request stats every second
|
||||||
statsInterval = setInterval(() => {
|
statsInterval = setInterval(() => {
|
||||||
if (window.activePeer) {
|
if (window.activePeer) {
|
||||||
const now = Date.now();
|
console.log('[INFO] Requesting container stats...');
|
||||||
if (now - lastStatsUpdate >= 500) { // Ensure at least 500ms between updates
|
sendCommand('stats', {}); // Adjust the command if specific arguments are needed
|
||||||
sendCommand('stats', {}); // Adjust command if necessary
|
|
||||||
lastStatsUpdate = now;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
console.warn('[WARN] No active peer; skipping stats request.');
|
console.warn('[WARN] No active peer; skipping stats request.');
|
||||||
}
|
}
|
||||||
}, 500); // Poll every 100ms for better reactivity
|
}, 1000); // 1 second interval
|
||||||
}
|
|
||||||
const smoothedStats = {}; // Container-specific smoothing storage
|
|
||||||
|
|
||||||
|
|
||||||
function smoothStats(containerId, newStats, smoothingFactor = 0.2) {
|
|
||||||
if (!smoothedStats[containerId]) {
|
|
||||||
smoothedStats[containerId] = { cpu: 0, memory: 0 };
|
|
||||||
}
|
|
||||||
|
|
||||||
smoothedStats[containerId].cpu =
|
|
||||||
smoothedStats[containerId].cpu * (1 - smoothingFactor) +
|
|
||||||
newStats.cpu * smoothingFactor;
|
|
||||||
|
|
||||||
smoothedStats[containerId].memory =
|
|
||||||
smoothedStats[containerId].memory * (1 - smoothingFactor) +
|
|
||||||
newStats.memory * smoothingFactor;
|
|
||||||
|
|
||||||
return smoothedStats[containerId];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function refreshContainerStats() {
|
function refreshContainerStats() {
|
||||||
@ -282,73 +261,43 @@ collapseSidebarBtn.addEventListener('click', () => {
|
|||||||
|
|
||||||
function handlePeerData(data, topicId, peer) {
|
function handlePeerData(data, topicId, peer) {
|
||||||
try {
|
try {
|
||||||
// Parse the incoming data
|
|
||||||
const response = JSON.parse(data.toString());
|
const response = JSON.parse(data.toString());
|
||||||
console.log(`[DEBUG] Received data from peer (topic: ${topicId}): ${JSON.stringify(response)}`);
|
console.log(`[DEBUG] Received data from peer (topic: ${topicId}): ${JSON.stringify(response)}`);
|
||||||
|
|
||||||
// Ensure the data is for the active connection
|
// Ensure the data is for the active connection
|
||||||
if (!connections[topicId]) {
|
if (!connections[topicId] || peer !== window.activePeer) {
|
||||||
console.warn(`[WARN] No connection found for topic: ${topicId}. Ignoring data.`);
|
console.warn(`[WARN] Ignoring data from inactive peer or topic: ${topicId}`);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (peer !== connections[topicId].peer) {
|
// Process the response based on its type
|
||||||
console.warn(`[WARN] Ignoring data from a non-active peer for topic: ${topicId}`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle errors in the response
|
|
||||||
if (response.error) {
|
if (response.error) {
|
||||||
console.error(`[ERROR] Server error received: ${response.error}`);
|
console.error(`[ERROR] Server error: ${response.error}`);
|
||||||
showAlert('danger', response.error);
|
showAlert('danger', response.error);
|
||||||
hideStatusIndicator();
|
hideStatusIndicator();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delegate handling based on the response type
|
if (response.type === 'containers') {
|
||||||
switch (response.type) {
|
renderContainers(response.data, topicId); // Scope containers to this topic
|
||||||
case 'containers':
|
} else if (response.type === 'stats') {
|
||||||
console.log('[INFO] Processing container list...');
|
|
||||||
renderContainers(response.data, topicId); // Render containers specific to this topic
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'stats':
|
|
||||||
console.log('[INFO] Updating container stats...');
|
|
||||||
if (!response.data.id) {
|
|
||||||
console.warn('[WARN] Stats response is missing container ID. Skipping update.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
response.data.topicId = topicId; // Attach the topicId to the stats
|
response.data.topicId = topicId; // Attach the topicId to the stats
|
||||||
updateContainerStats(response.data); // Update stats for the specified container
|
updateContainerStats(response.data); // Update stats for specific containers
|
||||||
break;
|
} else if (response.type === 'terminalOutput') {
|
||||||
|
|
||||||
case 'terminalOutput':
|
|
||||||
console.log('[INFO] Appending terminal output...');
|
|
||||||
appendTerminalOutput(response.data, response.containerId, response.encoding);
|
appendTerminalOutput(response.data, response.containerId, response.encoding);
|
||||||
break;
|
} else if (response.type === 'containerConfig') {
|
||||||
|
|
||||||
case 'containerConfig':
|
|
||||||
console.log('[INFO] Handling container configuration...');
|
|
||||||
if (window.inspectContainerCallback) {
|
if (window.inspectContainerCallback) {
|
||||||
window.inspectContainerCallback(response.data);
|
window.inspectContainerCallback(response.data);
|
||||||
window.inspectContainerCallback = null; // Reset the callback
|
window.inspectContainerCallback = null; // Reset the callback
|
||||||
}
|
}
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
console.warn(`[WARN] Unhandled response type: ${response.type}`);
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle peer response callback if defined
|
|
||||||
if (typeof window.handlePeerResponse === 'function') {
|
if (typeof window.handlePeerResponse === 'function') {
|
||||||
window.handlePeerResponse(response);
|
window.handlePeerResponse(response);
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
// Catch and log any parsing or processing errors
|
|
||||||
console.error(`[ERROR] Failed to process peer data: ${err.message}`);
|
console.error(`[ERROR] Failed to process peer data: ${err.message}`);
|
||||||
console.error(`[DEBUG] Raw data received: ${data.toString()}`);
|
showAlert('danger', 'Failed to process peer data.');
|
||||||
showAlert('danger', 'Failed to process peer data. Check the console for details.');
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -356,7 +305,6 @@ function handlePeerData(data, topicId, peer) {
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Add a new connection
|
// Add a new connection
|
||||||
addConnectionForm.addEventListener('submit', (e) => {
|
addConnectionForm.addEventListener('submit', (e) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
@ -398,7 +346,6 @@ function addConnection(topicHex) {
|
|||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
disconnectConnection(topicId, connectionItem);
|
disconnectConnection(topicId, connectionItem);
|
||||||
});
|
});
|
||||||
refreshContainerStats();
|
|
||||||
|
|
||||||
connectionList.appendChild(connectionItem);
|
connectionList.appendChild(connectionItem);
|
||||||
|
|
||||||
@ -423,13 +370,12 @@ function addConnection(topicHex) {
|
|||||||
window.activePeer = null;
|
window.activePeer = null;
|
||||||
dashboard.classList.add('hidden');
|
dashboard.classList.add('hidden');
|
||||||
containerList.innerHTML = '';
|
containerList.innerHTML = '';
|
||||||
stopStatsInterval(); // Stop stats polling
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!window.activePeer) {
|
if (!window.activePeer) {
|
||||||
switchConnection(topicId);
|
switchConnection(topicId);
|
||||||
}
|
}
|
||||||
startStatsInterval();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Collapse the sidebar after adding a connection
|
// Collapse the sidebar after adding a connection
|
||||||
@ -452,15 +398,6 @@ document.addEventListener('DOMContentLoaded', () => {
|
|||||||
const topicHex = savedConnections[topicId].topic;
|
const topicHex = savedConnections[topicId].topic;
|
||||||
addConnection(topicHex);
|
addConnection(topicHex);
|
||||||
});
|
});
|
||||||
|
|
||||||
if (Object.keys(connections).length > 0) {
|
|
||||||
hideWelcomePage();
|
|
||||||
startStatsInterval(); // Start stats polling for active peers
|
|
||||||
} else {
|
|
||||||
showWelcomePage();
|
|
||||||
}
|
|
||||||
|
|
||||||
assertVisibility();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
@ -832,33 +769,29 @@ function addActionListeners(row, container) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function updateStatsUI(row, stats) {
|
// Function to update container statistics
|
||||||
requestIdleCallback(() => {
|
function updateContainerStats(stats) {
|
||||||
|
console.log(`[DEBUG] Updating stats for container ID: ${stats.id}, Topic ID: ${stats.topicId}`);
|
||||||
|
|
||||||
|
// Ensure stats belong to the active connection
|
||||||
|
if (!window.activePeer || !connections[stats.topicId] || window.activePeer !== connections[stats.topicId].peer) {
|
||||||
|
console.warn(`[WARN] Stats received for inactive or unknown connection. Skipping.`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find the row for the container by its ID
|
||||||
|
const row = containerList.querySelector(`tr[data-container-id="${stats.id}"]`);
|
||||||
|
if (!row) {
|
||||||
|
console.warn(`[WARN] No matching row for container ID: ${stats.id}. Skipping stats update.`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the container statistics in the UI
|
||||||
row.querySelector('.cpu').textContent = stats.cpu.toFixed(2) || '0.00';
|
row.querySelector('.cpu').textContent = stats.cpu.toFixed(2) || '0.00';
|
||||||
row.querySelector('.memory').textContent = (stats.memory / (1024 * 1024)).toFixed(2) || '0.00';
|
row.querySelector('.memory').textContent = (stats.memory / (1024 * 1024)).toFixed(2) || '0.00';
|
||||||
row.querySelector('.ip-address').textContent = stats.ip || 'No IP Assigned';
|
row.querySelector('.ip-address').textContent = stats.ip || 'No IP Assigned';
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function updateContainerStats(stats) {
|
|
||||||
if (!stats || !stats.id || typeof stats.cpu === 'undefined' || typeof stats.memory === 'undefined') {
|
|
||||||
console.error('[ERROR] Invalid stats object:', stats);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`[DEBUG] Updating stats for container ID: ${stats.id}`);
|
|
||||||
|
|
||||||
const row = containerList.querySelector(`tr[data-container-id="${stats.id}"]`);
|
|
||||||
if (!row) {
|
|
||||||
console.warn(`[WARN] No matching row for container ID: ${stats.id}`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const smoothed = smoothStats(stats.id, stats);
|
|
||||||
updateStatsUI(row, smoothed);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Function to open the Duplicate Modal with container configurations
|
// Function to open the Duplicate Modal with container configurations
|
||||||
|
Loading…
x
Reference in New Issue
Block a user