mirror of
https://github.com/DeNNiiInc/Web-Page-Performance-Test.git
synced 2026-04-17 20:05:58 +00:00
134 lines
4.6 KiB
JavaScript
134 lines
4.6 KiB
JavaScript
const express = require("express");
|
|
const path = require("path");
|
|
const fs = require("fs");
|
|
const { exec } = require("child_process");
|
|
const runner = require("./lib/runner");
|
|
|
|
const app = express();
|
|
const PORT = process.env.PORT || 3000;
|
|
|
|
// Middleware
|
|
app.use(express.json());
|
|
app.use(express.static(__dirname));
|
|
app.use('/reports', express.static(path.join(__dirname, 'reports')));
|
|
|
|
// API Endpoint: Git Info
|
|
app.get("/api/git-info", (req, res) => {
|
|
exec('/usr/bin/git log -1 --format="%H|%cr"', (error, stdout, stderr) => {
|
|
if (error) {
|
|
console.error("Error getting git info:", error);
|
|
console.error("Stderr:", stderr);
|
|
return res.json({
|
|
commitId: "unknown",
|
|
commitAge: "dev mode",
|
|
error: true,
|
|
details: error.message
|
|
});
|
|
}
|
|
|
|
const [commitId, commitAge] = stdout.trim().split("|");
|
|
res.json({
|
|
commitId: commitId.substring(0, 7),
|
|
commitAge: commitAge,
|
|
error: false,
|
|
});
|
|
});
|
|
});
|
|
|
|
// API Endpoint: Run Test
|
|
app.post("/api/run-test", async (req, res) => {
|
|
const { url, isMobile, captureFilmstrip = false } = req.body;
|
|
const userUuid = req.headers['x-user-uuid'];
|
|
const userIp = req.headers['x-forwarded-for'] || req.socket.remoteAddress;
|
|
|
|
if (!url) return res.status(400).json({ error: "URL is required" });
|
|
|
|
try {
|
|
const result = await runner.runTest(url, { isMobile, userUuid, userIp, captureFilmstrip });
|
|
return res.json(result);
|
|
} catch (error) {
|
|
console.error("Test failed:", error);
|
|
res.status(500).json({ error: "Test failed", details: error.message });
|
|
}
|
|
});
|
|
|
|
// API Endpoint: History
|
|
app.get("/api/history", async (req, res) => {
|
|
const userUuid = req.headers['x-user-uuid'];
|
|
const userIp = req.headers['x-forwarded-for'] || req.socket.remoteAddress;
|
|
|
|
const history = await runner.getHistory(userUuid, userIp);
|
|
res.json(history);
|
|
});
|
|
|
|
// API Endpoint: Export HAR
|
|
app.get("/api/export/:testId/har", (req, res) => {
|
|
const { testId } = req.params;
|
|
const harPath = path.join(__dirname, 'reports', `${testId}.har.json`);
|
|
|
|
if (!fs.existsSync(harPath)) {
|
|
return res.status(404).json({ error: "HAR file not found" });
|
|
}
|
|
|
|
res.download(harPath, `test-${testId}.har`, (err) => {
|
|
if (err) console.error("Download error:", err);
|
|
});
|
|
});
|
|
|
|
// API Endpoint: Export CSV
|
|
app.get("/api/export/:testId/csv", (req, res) => {
|
|
const { testId } = req.params;
|
|
const jsonPath = path.join(__dirname, 'reports', `${testId}.json`);
|
|
|
|
if (!fs.existsSync(jsonPath)) {
|
|
return res.status(404).json({ error: "Test results not found" });
|
|
}
|
|
|
|
const data = JSON.parse(fs.readFileSync(jsonPath, 'utf8'));
|
|
const csv = `URL,Timestamp,Device,Performance,Accessibility,Best Practices,SEO,LCP,CLS,TBT
|
|
${data.url},${data.timestamp},${data.isMobile ? 'Mobile' : 'Desktop'},${data.scores.performance},${data.scores.accessibility},${data.scores.bestPractices},${data.scores.seo},${data.metrics.lcp},${data.metrics.cls},${data.metrics.tbt}`;
|
|
|
|
res.setHeader('Content-Type', 'text/csv');
|
|
res.setHeader('Content-Disposition', `attachment; filename="test-${testId}.csv"`);
|
|
res.send(csv);
|
|
});
|
|
|
|
// Serve index.html for all other routes
|
|
app.get("*", (req, res) => {
|
|
res.sendFile(path.join(__dirname, "index.html"));
|
|
});
|
|
|
|
// API Endpoint: Traceroute
|
|
app.get("/api/traceroute", (req, res) => {
|
|
const { host } = req.query;
|
|
if (!host) return res.status(400).json({ error: "Host required" });
|
|
|
|
// Sanitize host to prevent injection (basic)
|
|
if (/[^a-zA-Z0-9.-]/.test(host)) return res.status(400).json({ error: "Invalid host" });
|
|
|
|
const cmd = process.platform === 'win32' ? `tracert -h 10 ${host}` : `traceroute -m 10 ${host}`;
|
|
exec(cmd, (error, stdout, stderr) => {
|
|
res.json({ output: stdout || stderr });
|
|
});
|
|
});
|
|
|
|
// API Endpoint: Bulk Test
|
|
app.post("/api/bulk-test", async (req, res) => {
|
|
const { urls, isMobile, runCount = 1 } = req.body;
|
|
if (!urls || !Array.isArray(urls)) return res.status(400).json({ error: "URLs array required" });
|
|
|
|
// Mock response - in real world would queue these
|
|
// For now, we'll just acknowledge receipt. To implement fully requires a job queue.
|
|
// Let's implement a simple "Sequence" runner on the backend? No, that might timeout.
|
|
// Better to let frontend orchestrate or return a "Batch ID".
|
|
|
|
// We will assume frontend orchestration for simplicity in this Node.js (non-queue) env
|
|
res.json({ message: "Bulk test ready", count: urls.length });
|
|
});
|
|
|
|
app.listen(PORT, () => {
|
|
console.log(`🚀 Server running on port ${PORT}`);
|
|
console.log(`📁 Serving files from: ${__dirname}`);
|
|
console.log(`🌐 Environment: ${process.env.NODE_ENV || "development"}`);
|
|
});
|