mirror of
https://github.com/MrUnknownDE/utools.git
synced 2026-04-09 18:03:52 +02:00
166 lines
8.0 KiB
JavaScript
166 lines
8.0 KiB
JavaScript
// backend/routes/traceroute.js
|
|
const express = require('express');
|
|
const Sentry = require("@sentry/node");
|
|
const { spawn } = require('child_process');
|
|
const pino = require('pino');
|
|
|
|
// Import utilities
|
|
const { isValidIp, isPrivateIp, parseTracerouteLine } = require('../utils');
|
|
|
|
// Logger for this module
|
|
const logger = pino({ level: process.env.LOG_LEVEL || 'info' });
|
|
|
|
const router = express.Router();
|
|
|
|
// Helper function to safely create an error message string
|
|
function getErrorMessage(err, defaultMessage = 'An unknown error occurred') {
|
|
if (typeof err === 'string') return err;
|
|
if (err && typeof err.message === 'string' && err.message.trim() !== '') return err.message;
|
|
return defaultMessage;
|
|
}
|
|
|
|
|
|
// Route handler for / (relative to /api/traceroute)
|
|
router.get('/', (req, res) => {
|
|
const targetIpRaw = req.query.targetIp;
|
|
const targetIp = typeof targetIpRaw === 'string' ? targetIpRaw.trim() : targetIpRaw;
|
|
const requestIp = req.ip || req.socket.remoteAddress;
|
|
|
|
logger.info({ requestIp, targetIp }, 'Traceroute stream request received');
|
|
|
|
if (!isValidIp(targetIp)) {
|
|
logger.warn({ requestIp, targetIp }, 'Invalid target IP for traceroute');
|
|
return res.status(400).json({ success: false, error: 'Invalid target IP address provided.' });
|
|
}
|
|
if (isPrivateIp(targetIp)) {
|
|
logger.warn({ requestIp, targetIp }, 'Attempt to traceroute private IP blocked');
|
|
return res.status(403).json({ success: false, error: 'Operations on private or local IP addresses are not allowed.' });
|
|
}
|
|
|
|
// Add specific context for this request to the current Sentry scope
|
|
// Errors/Messages captured later in this request handler will have this context.
|
|
Sentry.configureScope(scope => {
|
|
scope.setContext("traceroute_details", { targetIp: targetIp, requestIp: requestIp });
|
|
});
|
|
|
|
|
|
try {
|
|
logger.info({ requestIp, targetIp }, `Starting traceroute stream...`);
|
|
res.setHeader('Content-Type', 'text/event-stream');
|
|
res.setHeader('Cache-Control', 'no-cache');
|
|
res.setHeader('Connection', 'keep-alive');
|
|
res.setHeader('X-Accel-Buffering', 'no');
|
|
res.flushHeaders();
|
|
|
|
const args = ['-n', targetIp];
|
|
const command = 'traceroute';
|
|
const proc = spawn(command, args);
|
|
logger.info({ requestIp, targetIp, command: `${command} ${args.join(' ')}` }, 'Spawned traceroute process');
|
|
|
|
let buffer = '';
|
|
|
|
const sendEvent = (event, data) => {
|
|
try {
|
|
if (!res.writableEnded) {
|
|
if (event === 'error' && (!data || typeof data.error !== 'string')) {
|
|
const safeErrorMessage = getErrorMessage(data?.error, 'Traceroute encountered an unspecified error.');
|
|
logger.warn({ requestIp, targetIp, originalData: data }, `Corrected invalid error event data. Sending: ${safeErrorMessage}`);
|
|
data = { error: safeErrorMessage };
|
|
}
|
|
res.write(`event: ${event}\ndata: ${JSON.stringify(data)}\n\n`);
|
|
} else {
|
|
logger.warn({ requestIp, targetIp, event }, "Attempted to write to closed SSE stream.");
|
|
}
|
|
} catch (e) {
|
|
// This catch handles errors during res.write, likely client disconnect
|
|
logger.error({ requestIp, targetIp, event, error: e.message }, "Error writing to SSE stream (client likely disconnected)");
|
|
Sentry.captureException(e, { level: 'warning', extra: { requestIp, targetIp, event } });
|
|
if (proc && !proc.killed) proc.kill();
|
|
if (!res.writableEnded) res.end();
|
|
// No manual transaction finishing needed here
|
|
}
|
|
};
|
|
|
|
proc.stdout.on('data', (data) => {
|
|
buffer += data.toString();
|
|
let lines = buffer.split('\n');
|
|
buffer = lines.pop() || '';
|
|
lines.forEach(line => {
|
|
const parsed = parseTracerouteLine(line);
|
|
if (parsed) {
|
|
// logger.debug({ requestIp, targetIp, hop: parsed.hop, ip: parsed.ip }, 'Sending hop data');
|
|
sendEvent('hop', parsed);
|
|
} else if (line.trim()) {
|
|
// logger.debug({ requestIp, targetIp, message: line.trim() }, 'Sending info data');
|
|
sendEvent('info', { message: line.trim() });
|
|
}
|
|
});
|
|
});
|
|
|
|
proc.stderr.on('data', (data) => {
|
|
const errorMsg = getErrorMessage(data.toString().trim(), 'Traceroute produced unknown stderr output.');
|
|
logger.warn({ requestIp, targetIp, stderr: errorMsg }, 'Traceroute stderr output');
|
|
Sentry.captureMessage('Traceroute stderr output', { level: 'warning', extra: { requestIp, targetIp, stderr: errorMsg } });
|
|
sendEvent('error', { error: errorMsg });
|
|
});
|
|
|
|
proc.on('error', (err) => {
|
|
const errorMsg = getErrorMessage(err, 'Failed to start traceroute command due to an unknown error.');
|
|
logger.error({ requestIp, targetIp, error: errorMsg }, `Failed to start traceroute command`);
|
|
Sentry.captureException(err, { extra: { requestIp, targetIp } }); // Capture original error
|
|
sendEvent('error', { error: `Failed to start traceroute: ${errorMsg}` });
|
|
if (!res.writableEnded) res.end();
|
|
// No manual transaction finishing needed here
|
|
});
|
|
|
|
proc.on('close', (code) => {
|
|
if (buffer) {
|
|
const parsed = parseTracerouteLine(buffer);
|
|
if (parsed) sendEvent('hop', parsed);
|
|
else if (buffer.trim()) sendEvent('info', { message: buffer.trim() });
|
|
}
|
|
|
|
if (code !== 0) {
|
|
const errorMsg = `Traceroute command failed with exit code ${code}`;
|
|
logger.error({ requestIp, targetIp, exitCode: code }, errorMsg);
|
|
Sentry.captureMessage('Traceroute command failed', { level: 'error', extra: { requestIp, targetIp, exitCode: code } });
|
|
sendEvent('error', { error: errorMsg });
|
|
// Transaction status will be inferred by Sentry based on errors captured
|
|
} else {
|
|
logger.info({ requestIp, targetIp }, `Traceroute stream completed successfully.`);
|
|
// Transaction status will likely be 'ok' if no errors were captured
|
|
}
|
|
sendEvent('end', { exitCode: code });
|
|
if (!res.writableEnded) res.end();
|
|
// No manual transaction finishing needed here
|
|
});
|
|
|
|
req.on('close', () => {
|
|
logger.info({ requestIp, targetIp }, 'Client disconnected from traceroute stream, killing process.');
|
|
if (proc && !proc.killed) proc.kill();
|
|
if (!res.writableEnded) res.end();
|
|
// Sentry transaction might be marked as 'cancelled' automatically or based on timeout
|
|
// No manual transaction finishing needed here
|
|
});
|
|
|
|
} catch (error) {
|
|
// This catch handles errors during the initial setup (e.g., spawn fails immediately)
|
|
const errorMsg = getErrorMessage(error, 'Failed to initiate traceroute due to an internal server error.');
|
|
logger.error({ requestIp, targetIp, error: errorMsg, stack: error.stack }, 'Error setting up traceroute stream');
|
|
Sentry.captureException(error, { extra: { requestIp, targetIp } }); // Capture original error
|
|
// No manual transaction finishing needed here
|
|
|
|
if (!res.headersSent) {
|
|
res.status(500).json({ success: false, error: `Failed to initiate traceroute: ${errorMsg}` });
|
|
} else {
|
|
try {
|
|
if (!res.writableEnded) {
|
|
sendEvent('error', { error: `Internal server error during setup: ${errorMsg}` });
|
|
res.end();
|
|
}
|
|
} catch (e) { logger.error({ requestIp, targetIp, error: e.message }, "Error writing final setup error to SSE stream"); }
|
|
}
|
|
}
|
|
});
|
|
|
|
module.exports = router; |