Skyvern/skyvern-frontend/artifactServer.js
Suchintan 43781af7c0
Add request logging and error handling to artifact and frontend servers (#4829)
Co-authored-by: Claude <noreply@anthropic.com>
2026-02-22 04:15:35 +00:00

93 lines
2.2 KiB
JavaScript

import express from "express";
import fs from "fs";
import cors from "cors";
const app = express();
app.use(cors());
// Request logging middleware — logs method, path, status, and duration
app.use((req, res, next) => {
const start = Date.now();
res.on("finish", () => {
const duration = Date.now() - start;
const timestamp = new Date().toISOString();
const artifactPath = req.query.path || "";
console.log(
"[%s] %s %s %d %dms %s",
timestamp,
req.method,
req.path,
res.statusCode,
duration,
artifactPath,
);
});
next();
});
app.get("/artifact/recording", (req, res) => {
const range = req.headers.range;
const path = req.query.path;
if (!path || !range) {
return res.status(400).send("Missing path or range header");
}
const videoSize = fs.statSync(path).size;
const chunkSize = 1 * 1e6;
const start = Number(range.replace(/\D/g, ""));
const end = Math.min(start + chunkSize, videoSize - 1);
const contentLength = end - start + 1;
const headers = {
"Content-Range": `bytes ${start}-${end}/${videoSize}`,
"Accept-Ranges": "bytes",
"Content-Length": contentLength,
"Content-Type": "video/mp4",
};
res.writeHead(206, headers);
const stream = fs.createReadStream(path, {
start,
end,
});
stream.pipe(res);
});
app.get("/artifact/image", (req, res) => {
const path = req.query.path;
res.sendFile(path);
});
app.get("/artifact/json", (req, res) => {
const path = req.query.path;
const contents = fs.readFileSync(path);
try {
const data = JSON.parse(contents);
res.json(data);
} catch (err) {
res.status(500).send(err);
}
});
app.get("/artifact/text", (req, res) => {
const path = req.query.path;
const contents = fs.readFileSync(path);
res.send(contents);
});
// Error handling middleware — catches unhandled errors in routes
app.use((err, req, res, _next) => {
const timestamp = new Date().toISOString();
console.error(
"[%s] ERROR %s %s:",
timestamp,
req.method,
req.path,
err.message,
);
res.status(500).send("Internal server error");
});
app.listen(9090, () => {
console.log(
`[${new Date().toISOString()}] Artifact server running at http://localhost:9090`,
);
});