222 lines
6.2 KiB
JavaScript
222 lines
6.2 KiB
JavaScript
const net = require('net');
|
|
const fs = require('fs');
|
|
const socketPath = '/tmp/oscar_watcher.sock';
|
|
|
|
//// UNIXSOCKET SERVER ////
|
|
|
|
let progvars = {};
|
|
|
|
// Remove the socket file if it already exists
|
|
if (fs.existsSync( socketPath )) {
|
|
try {
|
|
fs.unlinkSync(socketPath);
|
|
} catch (err) {
|
|
if (err.code !== 'ENOENT') {
|
|
console.error('Error removing existing socket file:', err);
|
|
process.exit(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
const unix_socket_server = net.createServer((connection) => {
|
|
console.log('UnixSocket Client connected.');
|
|
|
|
let buffer = '';
|
|
|
|
connection.on('data', (data) => {
|
|
buffer += data.toString();
|
|
let boundary = buffer.indexOf('\n');
|
|
while (boundary !== -1) {
|
|
const message = buffer.substring(0, boundary);
|
|
buffer = buffer.substring(boundary + 1);
|
|
processMessage(message);
|
|
boundary = buffer.indexOf('\n');
|
|
}
|
|
});
|
|
|
|
connection.on('end', () => {
|
|
console.log('UnixSocket Client disconnected.');
|
|
});
|
|
});
|
|
|
|
unix_socket_server.listen(socketPath, () => {
|
|
console.log(`UnixSocket Server listening on ${socketPath}`);
|
|
});
|
|
|
|
function processMessage(message) {
|
|
console.log(message);
|
|
try {
|
|
const { variable, value } = JSON.parse(message);
|
|
const diff = calculateDiff(progvars[variable], value);
|
|
send_ws_message(diff);
|
|
progvars[variable] = value;
|
|
|
|
const timestamp = new Date().toISOString();
|
|
const logEntry = `log_time="${timestamp}";\n${diff}\n`;
|
|
|
|
fs.writeFile('/data/openpilot/logs/watcher.log', logEntry, { flag: 'a' }, (err) => {
|
|
if (err) {
|
|
console.error('Error writing to log file:', err);
|
|
}
|
|
});
|
|
} catch (err) {
|
|
console.error('Error processing message:', err);
|
|
}
|
|
}
|
|
|
|
function calculateDiff(oldValue, newValue) {
|
|
const changes = [];
|
|
|
|
function findChanges(path, oldVal, newVal) {
|
|
const oldKeys = oldVal ? Object.keys(oldVal) : [];
|
|
const newKeys = Object.keys(newVal);
|
|
const allKeys = new Set([...oldKeys, ...newKeys]);
|
|
let changedCount = 0;
|
|
|
|
allKeys.forEach(key => {
|
|
const oldKeyValue = oldVal ? oldVal[key] : undefined;
|
|
const newKeyValue = newVal[key];
|
|
const currentPath = path ? `${path}['${key}']` : `['${key}']`;
|
|
|
|
if (!oldVal || !(key in oldVal)) {
|
|
// New key added
|
|
changes.push(`${currentPath} = ${JSON.stringify(newKeyValue)};`);
|
|
changedCount++;
|
|
} else if (typeof oldKeyValue === 'object' && oldKeyValue !== null && typeof newKeyValue === 'object') {
|
|
// Recursive diff for objects
|
|
const subChanges = findChanges(currentPath, oldKeyValue, newKeyValue);
|
|
if (subChanges > 0) changedCount++;
|
|
} else if (oldKeyValue !== newKeyValue) {
|
|
// Direct value change
|
|
changes.push(`${currentPath} = ${JSON.stringify(newKeyValue)};`);
|
|
changedCount++;
|
|
}
|
|
});
|
|
|
|
// If more than 1/3 of the properties have been changed, replace the entire node
|
|
if (changedCount > 0 && changedCount > oldKeys.length / 3) {
|
|
changes.push(`${path} = ${JSON.stringify(newVal)};`);
|
|
// Clear individual changes as we replace the entire node
|
|
return 0;
|
|
}
|
|
|
|
return changedCount;
|
|
}
|
|
|
|
findChanges('progvars', oldValue, newValue);
|
|
|
|
return changes.join('\n');
|
|
}
|
|
|
|
//// HTTP SERVER ////
|
|
|
|
const WebSocket = require('ws');
|
|
const http = require('http');
|
|
const url = require('url');
|
|
const path = require('path');
|
|
const hostname = '0.0.0.0';
|
|
const port = 1024;
|
|
|
|
const server = http.createServer((req, res) => {
|
|
if(req.url === '/') {
|
|
// Serve the HTML file
|
|
fs.readFile(path.join(__dirname, 'watcher.html'), (err, data) => {
|
|
if(err) {
|
|
res.writeHead(500);
|
|
res.end('Error loading watcher.html');
|
|
return;
|
|
}
|
|
res.writeHead(200, {'Content-Type': 'text/html'});
|
|
res.end(data);
|
|
});
|
|
} else {
|
|
// Handle 404
|
|
res.writeHead(404);
|
|
res.end('Not found');
|
|
}
|
|
});
|
|
|
|
server.listen(port, hostname, () => {
|
|
console.log(`Web Server running at http://${hostname}:${port}/`);
|
|
});
|
|
|
|
|
|
//// WEBSOCKET SERVER ////
|
|
ws_port = port + 1;
|
|
|
|
// Attach WebSocket server to the HTTP server
|
|
const wss = new WebSocket.Server({ port: ws_port });
|
|
|
|
console.log(`WebSocket Server running at http://${hostname}:${ws_port}/`);
|
|
|
|
wss.on('connection', function connection(ws) {
|
|
console.log('WebSocket client connected');
|
|
|
|
// Send initial state
|
|
ws.send(calculateDiff({}, progvars));
|
|
|
|
ws.on('message', function incoming(message) {
|
|
console.log('received: %s', message);
|
|
});
|
|
|
|
ws.on('close', function close() {
|
|
console.log('WebSocket client disconnected');
|
|
});
|
|
});
|
|
|
|
// Handle upgrade of the request
|
|
wss.on('upgrade', function upgrade(request, socket, head) {
|
|
const { pathname } = url.parse(request.url);
|
|
|
|
if (pathname === '/ws') {
|
|
wss.handleUpgrade(request, socket, head, function done(ws) {
|
|
wss.emit('connection', ws, request);
|
|
});
|
|
} else {
|
|
socket.destroy();
|
|
}
|
|
});
|
|
|
|
// Function to send a message to all connected WebSocket clients
|
|
function send_ws_message(message) {
|
|
wss.clients.forEach(function each(client) {
|
|
if (client.readyState === WebSocket.OPEN) {
|
|
client.send(message);
|
|
}
|
|
});
|
|
}
|
|
|
|
|
|
//// Exit when low disk space /////
|
|
|
|
const { exec } = require('child_process');
|
|
|
|
// Function to check disk space
|
|
function checkDiskSpace() {
|
|
exec('df -BG /data | tail -n 1 | awk \'{print $4}\'', (error, stdout, stderr) => {
|
|
if (error) {
|
|
console.error(`exec error: ${error}`);
|
|
return;
|
|
}
|
|
if (stderr) {
|
|
console.error(`stderr: ${stderr}`);
|
|
return;
|
|
}
|
|
|
|
// Extract the number of GBs available
|
|
const freeSpaceGB = parseInt(stdout.trim().replace('G', ''));
|
|
|
|
// Check if free space is less than 10GB
|
|
if (freeSpaceGB < 9) {
|
|
console.log('Less than 9GB of free space on /data. Exiting.');
|
|
process.exit(1);
|
|
}
|
|
});
|
|
}
|
|
|
|
// Check disk space every 1 minute
|
|
setInterval(checkDiskSpace, 60000);
|
|
|
|
// Perform an initial check
|
|
checkDiskSpace();
|