You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
59 lines
1.6 KiB
59 lines
1.6 KiB
import log4js from "logger";
|
|
import {
|
|
getRequestURL,
|
|
send,
|
|
setResponseHeader,
|
|
setResponseHeaders,
|
|
setResponseStatus,
|
|
} from "h3";
|
|
import { defineNitroErrorHandler } from "nitropack/runtime";
|
|
import { isJsonRequest, normalizeError } from "#server/utils/nitro-error-json";
|
|
|
|
const logger = log4js.getLogger("ERROR");
|
|
const isDev = process.env.NODE_ENV === "development";
|
|
|
|
/** 优先于 Nitro 内置 JSON 错误体:不向客户端返回 stack(开发/生产一致) */
|
|
export default defineNitroErrorHandler(function safeJsonErrorHandler(error, event) {
|
|
if (!isJsonRequest(event)) {
|
|
return;
|
|
}
|
|
|
|
const { statusCode, statusMessage, message } = normalizeError(error, isDev);
|
|
|
|
if (error.unhandled || error.fatal || statusCode >= 500) {
|
|
logger.error(
|
|
"[request]",
|
|
event.method,
|
|
event.path,
|
|
error.message,
|
|
error instanceof Error ? (error.stack ?? "") : "",
|
|
);
|
|
}
|
|
|
|
const url = getRequestURL(event, { xForwardedHost: true, xForwardedProto: true });
|
|
|
|
setResponseHeaders(event, {
|
|
"Content-Security-Policy": "script-src 'none'; frame-ancestors 'none';",
|
|
"X-Content-Type-Options": "nosniff",
|
|
"X-Frame-Options": "DENY",
|
|
"Referrer-Policy": "no-referrer",
|
|
});
|
|
if (statusCode === 404) {
|
|
setResponseHeader(event, "Cache-Control", "no-cache");
|
|
}
|
|
setResponseStatus(event, statusCode, statusMessage);
|
|
setResponseHeader(event, "Content-Type", "application/json");
|
|
|
|
const body: Record<string, unknown> = {
|
|
error: true,
|
|
url: url.href,
|
|
statusCode,
|
|
statusMessage,
|
|
message,
|
|
};
|
|
if (error.data !== undefined) {
|
|
body.data = error.data;
|
|
}
|
|
|
|
return send(event, JSON.stringify(body));
|
|
});
|
|
|