mirror of
https://github.com/nocobase/nocobase
synced 2024-11-15 18:16:31 +00:00
86 lines
2.5 KiB
TypeScript
86 lines
2.5 KiB
TypeScript
/**
|
|
* This file is part of the NocoBase (R) project.
|
|
* Copyright (c) 2020-2024 NocoBase Co., Ltd.
|
|
* Authors: NocoBase Team.
|
|
*
|
|
* This project is dual-licensed under AGPL-3.0 and NocoBase Commercial License.
|
|
* For more information, please refer to: https://www.nocobase.com/agreement.
|
|
*/
|
|
|
|
import { getLoggerFilePath } from './config';
|
|
import { Logger, LoggerOptions } from './logger';
|
|
import { pick } from 'lodash';
|
|
const defaultRequestWhitelist = [
|
|
'action',
|
|
'header.x-role',
|
|
'header.x-hostname',
|
|
'header.x-timezone',
|
|
'header.x-locale',
|
|
'header.x-authenticator',
|
|
'header.x-data-source',
|
|
'referer',
|
|
];
|
|
const defaultResponseWhitelist = ['status'];
|
|
|
|
export interface RequestLoggerOptions extends LoggerOptions {
|
|
skip?: (ctx?: any) => Promise<boolean>;
|
|
requestWhitelist?: string[];
|
|
responseWhitelist?: string[];
|
|
}
|
|
|
|
export const requestLogger = (appName: string, requestLogger: Logger, options?: RequestLoggerOptions) => {
|
|
return async (ctx, next) => {
|
|
const reqId = ctx.reqId;
|
|
const path = /^\/api\/(.+):(.+)/.exec(ctx.path);
|
|
const contextLogger = ctx.app.log.child({ reqId, module: path?.[1], submodule: path?.[2] });
|
|
// ctx.reqId = reqId;
|
|
ctx.logger = ctx.log = contextLogger;
|
|
const startTime = Date.now();
|
|
const requestInfo = {
|
|
method: ctx.method,
|
|
path: ctx.url,
|
|
};
|
|
requestLogger.info({
|
|
message: `request ${ctx.method} ${ctx.url}`,
|
|
...requestInfo,
|
|
req: pick(ctx.request.toJSON(), options?.requestWhitelist || defaultRequestWhitelist),
|
|
action: ctx.action?.toJSON?.(),
|
|
app: appName,
|
|
reqId,
|
|
});
|
|
let error: Error;
|
|
try {
|
|
await next();
|
|
} catch (e) {
|
|
error = e;
|
|
} finally {
|
|
const cost = Date.now() - startTime;
|
|
const status = ctx.status;
|
|
const info = {
|
|
message: `response ${ctx.url}`,
|
|
...requestInfo,
|
|
res: pick(ctx.response.toJSON(), options?.responseWhitelist || defaultResponseWhitelist),
|
|
action: ctx.action?.toJSON?.(),
|
|
userId: ctx.auth?.user?.id,
|
|
status: ctx.status,
|
|
cost,
|
|
app: appName,
|
|
reqId,
|
|
};
|
|
if (Math.floor(status / 100) == 5) {
|
|
requestLogger.error({ ...info, res: ctx.body?.['errors'] || ctx.body });
|
|
} else if (Math.floor(status / 100) == 4) {
|
|
requestLogger.warn({ ...info, res: ctx.body?.['errors'] || ctx.body });
|
|
} else {
|
|
requestLogger.info(info);
|
|
}
|
|
}
|
|
|
|
ctx.res.setHeader('X-Request-Id', reqId);
|
|
|
|
if (error) {
|
|
throw error;
|
|
}
|
|
};
|
|
};
|