peartainer/server/server.js

430 lines
15 KiB
JavaScript
Raw Normal View History

2024-11-26 08:14:43 -05:00
// server.js
2024-11-26 03:36:51 -05:00
import Hyperswarm from 'hyperswarm';
import Docker from 'dockerode';
import crypto from 'hypercore-crypto';
2024-11-28 02:51:47 -05:00
import { PassThrough } from 'stream';
2024-11-26 03:36:51 -05:00
const docker = new Docker({ socketPath: '/var/run/docker.sock' });
const swarm = new Hyperswarm();
const connectedPeers = new Set();
2024-11-26 08:14:43 -05:00
const terminalSessions = new Map(); // Map to track terminal sessions per peer
2024-11-26 03:36:51 -05:00
// Generate a topic for the server
const topic = crypto.randomBytes(32);
console.log(`[INFO] Server started with topic: ${topic.toString('hex')}`);
2024-11-26 08:14:43 -05:00
// Join the swarm with the generated topic
2024-11-26 03:36:51 -05:00
swarm.join(topic, { server: true, client: false });
2024-11-26 08:14:43 -05:00
// Handle incoming peer connections
2024-11-26 03:36:51 -05:00
swarm.on('connection', (peer) => {
2024-11-26 08:14:43 -05:00
console.log('[INFO] Peer connected');
2024-11-26 03:36:51 -05:00
connectedPeers.add(peer);
peer.on('data', async (data) => {
try {
const parsedData = JSON.parse(data.toString());
console.log(`[DEBUG] Received data from peer: ${JSON.stringify(parsedData)}`);
let response;
switch (parsedData.command) {
case 'listContainers':
2024-11-26 08:14:43 -05:00
console.log('[INFO] Handling \'listContainers\' command');
2024-11-26 03:36:51 -05:00
const containers = await docker.listContainers({ all: true });
response = { type: 'containers', data: containers };
break;
2024-11-26 08:14:43 -05:00
case 'inspectContainer':
console.log(`[INFO] Handling 'inspectContainer' command for container: ${parsedData.args.id}`);
const container = docker.getContainer(parsedData.args.id);
const config = await container.inspect();
response = { type: 'containerConfig', data: config };
break;
case 'duplicateContainer':
console.log('[INFO] Handling \'duplicateContainer\' command');
2024-11-28 07:07:46 -05:00
const { name, image, hostname, netmode, cpu, memory, config: dupConfig } = parsedData.args;
const memoryInMB = memory * 1024 * 1024;
console.log("MEMEMMEMEMEMEMEMEMMEME " + memoryInMB)
await duplicateContainer(name, image, hostname, netmode, cpu, memoryInMB, dupConfig, peer);
2024-11-26 08:14:43 -05:00
return; // Response is handled within the duplicateContainer function
2024-11-26 03:36:51 -05:00
case 'startContainer':
console.log(`[INFO] Handling 'startContainer' command for container: ${parsedData.args.id}`);
await docker.getContainer(parsedData.args.id).start();
response = { success: true, message: `Container ${parsedData.args.id} started` };
break;
case 'stopContainer':
console.log(`[INFO] Handling 'stopContainer' command for container: ${parsedData.args.id}`);
await docker.getContainer(parsedData.args.id).stop();
response = { success: true, message: `Container ${parsedData.args.id} stopped` };
break;
case 'removeContainer':
console.log(`[INFO] Handling 'removeContainer' command for container: ${parsedData.args.id}`);
await docker.getContainer(parsedData.args.id).remove({ force: true });
response = { success: true, message: `Container ${parsedData.args.id} removed` };
break;
case 'startTerminal':
console.log(`[INFO] Starting terminal for container: ${parsedData.args.containerId}`);
handleTerminal(parsedData.args.containerId, peer);
2024-11-26 08:14:43 -05:00
return; // No immediate response needed for streaming commands
case 'killTerminal':
console.log(`[INFO] Handling 'killTerminal' command for container: ${parsedData.args.containerId}`);
handleKillTerminal(parsedData.args.containerId, peer);
response = {
success: true,
message: `Terminal for container ${parsedData.args.containerId} killed`,
};
break;
2024-11-26 03:36:51 -05:00
default:
2024-11-28 02:51:47 -05:00
// console.warn(`[WARN] Unknown command: ${parsedData.command}`);
// response = { error: 'Unknown command' };
return
2024-11-26 03:36:51 -05:00
}
2024-11-26 08:14:43 -05:00
// Send response if one was generated
if (response) {
console.log(`[DEBUG] Sending response to peer: ${JSON.stringify(response)}`);
peer.write(JSON.stringify(response));
}
2024-11-26 03:36:51 -05:00
} catch (err) {
console.error(`[ERROR] Failed to handle data from peer: ${err.message}`);
peer.write(JSON.stringify({ error: err.message }));
}
});
2024-11-28 02:51:47 -05:00
peer.on('error', (err) => {
console.error(`[ERROR] Peer connection error: ${err.message}`);
cleanupPeer(peer);
});
2024-11-26 03:36:51 -05:00
peer.on('close', () => {
2024-11-26 08:14:43 -05:00
console.log('[INFO] Peer disconnected');
2024-11-26 03:36:51 -05:00
connectedPeers.delete(peer);
2024-11-28 02:51:47 -05:00
cleanupPeer(peer)
2024-11-26 08:14:43 -05:00
// Clean up any terminal session associated with this peer
if (terminalSessions.has(peer)) {
const session = terminalSessions.get(peer);
console.log(`[INFO] Cleaning up terminal session for container: ${session.containerId}`);
session.stream.end();
peer.removeListener('data', session.onData);
terminalSessions.delete(peer);
}
2024-11-26 03:36:51 -05:00
});
});
2024-11-28 02:51:47 -05:00
// Helper function to handle peer cleanup
function cleanupPeer(peer) {
connectedPeers.delete(peer);
if (terminalSessions.has(peer)) {
const session = terminalSessions.get(peer);
console.log(`[INFO] Cleaning up terminal session for container: ${session.containerId}`);
session.stream.end();
peer.removeListener('data', session.onData);
terminalSessions.delete(peer);
}
}
2024-11-26 08:14:43 -05:00
// Function to duplicate a container
2024-11-28 07:07:46 -05:00
async function duplicateContainer(name, image, hostname, netmode, cpu, memory, config, peer) {
2024-11-26 08:14:43 -05:00
try {
// Remove non-essential fields from the configuration
const sanitizedConfig = { ...config };
delete sanitizedConfig.Id;
delete sanitizedConfig.State;
delete sanitizedConfig.Created;
delete sanitizedConfig.NetworkSettings;
delete sanitizedConfig.Mounts;
delete sanitizedConfig.Path;
delete sanitizedConfig.Args;
delete sanitizedConfig.Image;
2024-11-28 07:07:46 -05:00
delete sanitizedConfig.Hostname;
delete sanitizedConfig.CpuCount;
delete sanitizedConfig.Memory;
delete sanitizedConfig.CpuShares;
delete sanitizedConfig.CpusetCpus;
2024-11-26 08:14:43 -05:00
// Ensure the container has a unique name
const newName = name;
const existingContainers = await docker.listContainers({ all: true });
const nameExists = existingContainers.some(c => c.Names.includes(`/${newName}`));
if (nameExists) {
peer.write(JSON.stringify({ error: `Container name '${newName}' already exists.` }));
return;
}
2024-11-28 07:07:46 -05:00
const cpusetCpus = Array.from({ length: cpu }, (_, i) => i).join(",");
const nanoCpus = cpu * 1e9;
2024-11-26 08:14:43 -05:00
// Create a new container with the provided configuration
const newContainer = await docker.createContainer({
2024-11-28 07:07:46 -05:00
...sanitizedConfig.Config, // General configuration
name: newName, // Container name
Hostname: hostname, // Hostname for the container
Image: image, // Container image
HostConfig: { // Host-specific configurations
CpusetCpus: cpusetCpus.toString(), // Number of CPUs
NanoCpus: nanoCpus, // Restrict CPU time (e.g., 4 cores = 4e9 nanoseconds)
Memory: Number(memory), // Memory limit in bytes
MemoryReservation: Number(memory), // Memory limit in bytes
NetworkMode: netmode.toString(), // Network mode
},
2024-11-26 08:14:43 -05:00
});
// Start the new container
await newContainer.start();
2024-11-28 02:51:47 -05:00
// Send success response to the requesting peer
2024-11-26 08:14:43 -05:00
peer.write(JSON.stringify({ success: true, message: `Container '${newName}' duplicated and started successfully.` }));
2024-11-28 02:51:47 -05:00
// Get the updated list of containers
2024-11-26 08:14:43 -05:00
const containers = await docker.listContainers({ all: true });
const update = { type: 'containers', data: containers };
2024-11-28 02:51:47 -05:00
// Broadcast the updated container list to all connected peers
2024-11-26 08:14:43 -05:00
for (const connectedPeer of connectedPeers) {
connectedPeer.write(JSON.stringify(update));
}
2024-11-28 02:51:47 -05:00
// Start streaming stats for the new container
const newContainerInfo = containers.find(c => c.Names.includes(`/${newName}`));
if (newContainerInfo) {
streamContainerStats(newContainerInfo);
}
2024-11-26 08:14:43 -05:00
} catch (err) {
console.error(`[ERROR] Failed to duplicate container: ${err.message}`);
peer.write(JSON.stringify({ error: `Failed to duplicate container: ${err.message}` }));
}
}
2024-11-28 02:51:47 -05:00
2024-11-26 03:36:51 -05:00
// Stream Docker events to all peers
docker.getEvents({}, (err, stream) => {
if (err) {
console.error(`[ERROR] Failed to get Docker events: ${err.message}`);
return;
}
stream.on('data', async (chunk) => {
try {
const event = JSON.parse(chunk.toString());
2024-11-26 08:14:43 -05:00
if (event.status === "undefined") return
2024-11-26 03:36:51 -05:00
console.log(`[INFO] Docker event received: ${event.status} - ${event.id}`);
2024-11-26 08:14:43 -05:00
// Get updated container list and broadcast it to all connected peers
2024-11-26 03:36:51 -05:00
const containers = await docker.listContainers({ all: true });
const update = { type: 'containers', data: containers };
for (const peer of connectedPeers) {
peer.write(JSON.stringify(update));
}
} catch (err) {
console.error(`[ERROR] Failed to process Docker event: ${err.message}`);
}
});
});
2024-11-26 08:14:43 -05:00
// Collect and stream container stats
docker.listContainers({ all: true }, (err, containers) => {
if (err) {
console.error(`[ERROR] Failed to list containers for stats: ${err.message}`);
return;
}
containers.forEach((containerInfo) => {
const container = docker.getContainer(containerInfo.Id);
container.stats({ stream: true }, (err, stream) => {
if (err) {
return;
}
stream.on('data', (data) => {
try {
const stats = JSON.parse(data.toString());
const cpuUsage = calculateCPUPercent(stats);
const memoryUsage = stats.memory_stats.usage;
const networks = stats.networks;
const ipAddress = networks ? Object.values(networks)[0].IPAddress : '-';
const statsData = {
id: containerInfo.Id,
cpu: cpuUsage,
memory: memoryUsage,
ip: ipAddress,
};
// Broadcast stats to all connected peers
for (const peer of connectedPeers) {
peer.write(JSON.stringify({ type: 'stats', data: statsData }));
}
} catch (err) {
}
});
stream.on('error', (err) => {
});
});
});
});
// Function to calculate CPU usage percentage
function calculateCPUPercent(stats) {
const cpuDelta = stats.cpu_stats.cpu_usage.total_usage - stats.precpu_stats.cpu_usage.total_usage;
const systemDelta = stats.cpu_stats.system_cpu_usage - stats.precpu_stats.system_cpu_usage;
const cpuCount = stats.cpu_stats.online_cpus || stats.cpu_stats.cpu_usage.percpu_usage.length;
if (systemDelta > 0.0 && cpuDelta > 0.0) {
return (cpuDelta / systemDelta) * cpuCount * 100.0;
}
return 0.0;
}
2024-11-28 02:51:47 -05:00
// Function to handle terminal sessions
2024-11-26 08:14:43 -05:00
// Function to handle terminal sessions
2024-11-26 03:36:51 -05:00
async function handleTerminal(containerId, peer) {
const container = docker.getContainer(containerId);
try {
const exec = await container.exec({
2024-11-26 08:14:43 -05:00
Cmd: ['/bin/bash'],
2024-11-26 03:36:51 -05:00
AttachStdin: true,
AttachStdout: true,
AttachStderr: true,
Tty: true,
});
const stream = await exec.start({ hijack: true, stdin: true });
console.log(`[INFO] Terminal session started for container: ${containerId}`);
2024-11-28 02:51:47 -05:00
const stdout = new PassThrough();
const stderr = new PassThrough();
container.modem.demuxStream(stream, stdout, stderr);
2024-11-26 08:14:43 -05:00
const onData = (input) => {
2024-11-26 03:36:51 -05:00
try {
const parsed = JSON.parse(input.toString());
if (parsed.type === 'terminalInput' && parsed.data) {
2024-11-28 02:51:47 -05:00
const inputData = parsed.encoding === 'base64'
? Buffer.from(parsed.data, 'base64')
: Buffer.from(parsed.data);
2024-11-26 08:14:43 -05:00
stream.write(inputData);
2024-11-26 03:36:51 -05:00
}
} catch (err) {
console.error(`[ERROR] Failed to parse terminal input: ${err.message}`);
}
2024-11-26 08:14:43 -05:00
};
peer.on('data', onData);
terminalSessions.set(peer, { containerId, exec, stream, onData });
2024-11-28 02:51:47 -05:00
stdout.on('data', (chunk) => {
2024-11-26 08:14:43 -05:00
peer.write(JSON.stringify({
type: 'terminalOutput',
containerId,
2024-11-28 02:51:47 -05:00
data: chunk.toString('base64'),
encoding: 'base64',
}));
});
stderr.on('data', (chunk) => {
peer.write(JSON.stringify({
type: 'terminalErrorOutput',
containerId,
data: chunk.toString('base64'),
2024-11-26 08:14:43 -05:00
encoding: 'base64',
}));
2024-11-26 03:36:51 -05:00
});
peer.on('close', () => {
console.log(`[INFO] Peer disconnected, ending terminal session for container: ${containerId}`);
stream.end();
2024-11-26 08:14:43 -05:00
terminalSessions.delete(peer);
peer.removeListener('data', onData);
2024-11-26 03:36:51 -05:00
});
} catch (err) {
console.error(`[ERROR] Failed to start terminal for container ${containerId}: ${err.message}`);
peer.write(JSON.stringify({ error: `Failed to start terminal: ${err.message}` }));
}
}
2024-11-28 02:51:47 -05:00
2024-11-26 08:14:43 -05:00
// Function to handle killing terminal sessions
function handleKillTerminal(containerId, peer) {
const session = terminalSessions.get(peer);
if (session && session.containerId === containerId) {
console.log(`[INFO] Killing terminal session for container: ${containerId}`);
// Close the stream and exec session
session.stream.end();
terminalSessions.delete(peer);
// Remove the specific 'data' event listener for terminal input
peer.removeListener('data', session.onData);
console.log(`[INFO] Terminal session for container ${containerId} terminated`);
} else {
console.warn(`[WARN] No terminal session found for container: ${containerId}`);
}
}
2024-11-28 02:51:47 -05:00
function streamContainerStats(containerInfo) {
const container = docker.getContainer(containerInfo.Id);
container.stats({ stream: true }, (err, stream) => {
if (err) {
console.error(`[ERROR] Failed to get stats for container ${containerInfo.Id}: ${err.message}`);
return;
}
stream.on('data', (data) => {
try {
const stats = JSON.parse(data.toString());
const cpuUsage = calculateCPUPercent(stats);
const memoryUsage = stats.memory_stats.usage;
const networks = stats.networks;
const ipAddress = networks ? Object.values(networks)[0].IPAddress : '-';
const statsData = {
id: containerInfo.Id,
cpu: cpuUsage,
memory: memoryUsage,
ip: ipAddress,
};
// Broadcast stats to all connected peers
for (const peer of connectedPeers) {
peer.write(JSON.stringify({ type: 'stats', data: statsData }));
}
} catch (err) {
console.error(`[ERROR] Failed to parse stats for container ${containerInfo.Id}: ${err.message}`);
}
});
stream.on('error', (err) => {
console.error(`[ERROR] Stats stream error for container ${containerInfo.Id}: ${err.message}`);
});
});
}
2024-11-26 08:14:43 -05:00
// Handle process termination
2024-11-26 03:36:51 -05:00
process.on('SIGINT', () => {
2024-11-26 08:14:43 -05:00
console.log('[INFO] Server shutting down');
2024-11-26 03:36:51 -05:00
swarm.destroy();
process.exit();
2024-11-28 07:07:46 -05:00
});