agent-smith/dist-in/commons/websocket.js
2026-02-26 19:41:09 +01:00

228 lines
20 KiB
JavaScript

import { WebSocketServer, WebSocket } from 'ws';
import fs from 'fs';
import path from 'path';
import chokidar from 'chokidar';
export class WebSocketManager {
static instance;
wss = null;
handlers = new Map();
writeQueue = Promise.resolve();
constructor() {
// Register default handlers
this.registerHandler('log', this.handleLog.bind(this));
this.registerHandler('echo', (ws, payload) => ws.send(JSON.stringify({ type: 'echo', payload })));
this.registerHandler('ping', (ws, payload) => ws.send(JSON.stringify({ type: 'pong', id: payload.id })));
}
static getInstance() {
if (!WebSocketManager.instance) {
WebSocketManager.instance = new WebSocketManager();
}
return WebSocketManager.instance;
}
init(server) {
if (this.wss) {
console.warn('WebSocketServer already initialized');
return;
}
this.wss = new WebSocketServer({ server, path: '/ws' });
this.wss.on('connection', (ws) => {
ws.on('message', (message) => {
try {
const data = JSON.parse(message.toString());
const { command, ...payload } = data;
if (command && this.handlers.has(command)) {
this.handlers.get(command)(ws, payload);
}
else {
console.warn('Unknown command:', command);
}
}
catch (err) {
console.error('Failed to parse message:', err);
}
});
ws.on('close', () => {
});
ws.on('error', (err) => {
console.error('WebSocket error:', err);
});
});
this.initWatcher();
}
initWatcher() {
// Watch for changes in canvas-page-new.json
const logDir = path.join(process.cwd(), 'data');
// Ensure log directory exists
if (!fs.existsSync(logDir)) {
try {
fs.mkdirSync(logDir, { recursive: true });
}
catch (err) {
console.error('Failed to create log directory for watcher:', err);
}
}
const handleFile = async (filePath) => {
// Ignore output files (logs) to prevent infinite loops (Frontend -> Log -> Watcher -> Frontend -> Loop)
const fileName = path.basename(filePath);
const ext = path.extname(filePath).toLowerCase();
// Explicitly allow only specific JSON files (layouts) to trigger updates
// Ignore everything else (logs, dumps, etc.)
if (ext === '.json') {
if (fileName !== 'canvas-page-latest-new.json' && fileName !== 'canvas-page-new.json') {
return;
}
}
else if (fileName.startsWith('canvas-html-latest')) {
return;
}
console.log(`[Watcher] File detected: ${filePath}`);
try {
const ext = path.extname(filePath).toLowerCase();
if (ext === '.json') {
const content = await fs.promises.readFile(filePath, 'utf-8');
if (!content.trim())
return; // Ignore empty writes
try {
const layoutData = JSON.parse(content);
console.log('Broadcasting layout-update (json)...');
this.broadcast({
type: 'layout-update',
data: layoutData
});
}
catch (parseErr) {
console.error(`Failed to parse watched JSON file: ${filePath}`, parseErr);
}
}
else if (ext === '.html' || ext === '.md') {
const content = await fs.promises.readFile(filePath, 'base64');
console.log(`Broadcasting layout-update (${ext})...`);
this.broadcast({
type: 'layout-update',
data: content
});
}
}
catch (err) {
console.error(`Failed to process watched file ${filePath}:`, err);
}
};
chokidar.watch(logDir, {
persistent: true,
ignoreInitial: false,
awaitWriteFinish: {
stabilityThreshold: 100,
pollInterval: 100
}
})
.on('add', handleFile)
.on('change', handleFile);
}
registerHandler(command, handler) {
this.handlers.set(command, handler);
}
broadcast(message) {
if (!this.wss)
return;
const data = JSON.stringify(message);
this.wss.clients.forEach((client) => {
if (client.readyState === WebSocket.OPEN) {
client.send(data);
}
});
}
handleLog(ws, payload) {
// Expected payload: { name: string, options?: { mode?: 'append'|'overwrite', format?: 'json'|'html'|'md' }, message: any, ...others }
const { name, id, options, ...logData } = payload;
if (!name) {
console.warn('Log command missing "name" field');
return;
}
const mode = options?.mode || 'append';
const format = options?.format || 'json';
const logDir = path.join(process.cwd(), 'data');
const extension = format === 'md' ? 'md' : format === 'html' ? 'html' : 'json';
const logFile = path.join(logDir, `${name}.${extension}`);
// Ensure log directory exists
if (!fs.existsSync(logDir)) {
try {
fs.mkdirSync(logDir, { recursive: true });
}
catch (err) {
console.error('Failed to create log directory:', err);
return;
}
}
// Serialize writes using the queue
this.writeQueue = this.writeQueue.then(async () => {
try {
if (format === 'json') {
if (mode === 'overwrite') {
// For overwrite (state capture), write only the message content if available
const content = (logData.message !== undefined) ? logData.message : logData;
const contentToWrite = JSON.stringify(content, null, 2);
await fs.promises.writeFile(logFile, contentToWrite);
}
else {
// For append (logging), read existing, parse, append to array, write back
let records = [];
try {
if (fs.existsSync(logFile)) {
const fileContent = await fs.promises.readFile(logFile, 'utf-8');
if (fileContent.trim()) {
try {
const parsed = JSON.parse(fileContent);
if (Array.isArray(parsed)) {
records = parsed;
}
else {
records = [parsed];
}
}
catch (e) {
// Attempt to parse as NDJSON (newline delimited JSON)
records = fileContent.split('\n')
.filter(line => line.trim())
.map(line => {
try {
return JSON.parse(line);
}
catch {
return null;
}
})
.filter(item => item !== null);
}
}
}
}
catch (readErr) {
console.warn(`Failed to read log file ${logFile}, starting fresh.`, readErr);
}
const logEntry = {
timestamp: new Date().toISOString(),
...logData
};
records.push(logEntry);
await fs.promises.writeFile(logFile, JSON.stringify(records, null, 2));
}
}
else {
// HTML or MD
const message = logData.message;
const content = typeof message === 'string' ? message : JSON.stringify(message);
if (mode === 'append') {
await fs.promises.appendFile(logFile, content + '\n');
}
else {
await fs.promises.writeFile(logFile, content);
}
}
}
catch (err) {
console.error(`Failed to write log file ${logFile}:`, err);
}
});
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket.js","sourceRoot":"","sources":["../../src/commons/websocket.ts"],"names":[],"mappings":"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"}