0
Fork 0
mirror of https://github.com/logto-io/logto.git synced 2025-01-20 21:32:31 -05:00

feat(core): log error body (#1065)

This commit is contained in:
IceHe.xyz 2022-06-07 18:16:14 +08:00 committed by GitHub
parent b2b71898d3
commit 2ba11215ed
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 77 additions and 30 deletions

View file

@ -1,5 +1,7 @@
import { LogPayload, LogResult } from '@logto/schemas'; import { LogPayload, LogResult } from '@logto/schemas';
import i18next from 'i18next';
import RequestError from '@/errors/RequestError';
import { insertLog } from '@/queries/log'; import { insertLog } from '@/queries/log';
import { createContextWithRouteParameters } from '@/utils/test-utils'; import { createContextWithRouteParameters } from '@/utils/test-utils';
@ -21,7 +23,7 @@ jest.mock('nanoid', () => ({
describe('koaLog middleware', () => { describe('koaLog middleware', () => {
const insertLogMock = insertLog as jest.Mock; const insertLogMock = insertLog as jest.Mock;
const type = 'SignInUsernamePassword'; const type = 'SignInUsernamePassword';
const payload: LogPayload = { const mockPayload: LogPayload = {
userId: 'foo', userId: 'foo',
username: 'Bar', username: 'Bar',
}; };
@ -34,7 +36,7 @@ describe('koaLog middleware', () => {
jest.clearAllMocks(); jest.clearAllMocks();
}); });
it('insert log with success response', async () => { it('should insert a success log when next() does not throw an error', async () => {
const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = { const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = {
...createContextWithRouteParameters({ headers: { 'user-agent': userAgent } }), ...createContextWithRouteParameters({ headers: { 'user-agent': userAgent } }),
// Bypass middleware context type assert // Bypass middleware context type assert
@ -44,7 +46,7 @@ describe('koaLog middleware', () => {
ctx.request.ip = ip; ctx.request.ip = ip;
const next = async () => { const next = async () => {
ctx.log(type, payload); ctx.log(type, mockPayload);
}; };
await koaLog()(ctx, next); await koaLog()(ctx, next);
@ -52,7 +54,7 @@ describe('koaLog middleware', () => {
id: nanoIdMock, id: nanoIdMock,
type, type,
payload: { payload: {
...payload, ...mockPayload,
result: LogResult.Success, result: LogResult.Success,
ip, ip,
userAgent, userAgent,
@ -60,33 +62,70 @@ describe('koaLog middleware', () => {
}); });
}); });
it('should insert log with failed result if next throws error', async () => { describe('should insert an error log with the error message when next() throws an error', () => {
const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = { it('should log with error message when next throws a normal Error', async () => {
...createContextWithRouteParameters({ headers: { 'user-agent': userAgent } }), const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = {
// Bypass middleware context type assert ...createContextWithRouteParameters({ headers: { 'user-agent': userAgent } }),
addLogContext, // Bypass middleware context type assert
log, addLogContext,
}; log,
ctx.request.ip = ip; };
ctx.request.ip = ip;
const error = new Error('next error'); const message = 'Normal error';
const error = new Error(message);
const next = async () => { const next = async () => {
ctx.log(type, payload); ctx.log(type, mockPayload);
throw error; throw error;
}; };
await expect(koaLog()(ctx, next)).rejects.toMatchError(error); await expect(koaLog()(ctx, next)).rejects.toMatchError(error);
expect(insertLogMock).toBeCalledWith({ expect(insertLogMock).toBeCalledWith({
id: nanoIdMock, id: nanoIdMock,
type, type,
payload: { payload: {
...payload, ...mockPayload,
result: LogResult.Error, result: LogResult.Error,
error: String(error), error: { message: `Error: ${message}` },
ip, ip,
userAgent, userAgent,
}, },
});
});
it('should insert an error log with the error body when next() throws a RequestError', async () => {
const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = {
...createContextWithRouteParameters({ headers: { 'user-agent': userAgent } }),
// Bypass middleware context type assert
addLogContext,
log,
};
ctx.request.ip = ip;
const message = 'Error message';
jest.spyOn(i18next, 't').mockReturnValueOnce(message); // Used in
const code = 'connector.general';
const data = { foo: 'bar', num: 123 };
const error = new RequestError(code, data);
const next = async () => {
ctx.log(type, mockPayload);
throw error;
};
await expect(koaLog()(ctx, next)).rejects.toMatchError(error);
expect(insertLogMock).toBeCalledWith({
id: nanoIdMock,
type,
payload: {
...mockPayload,
result: LogResult.Error,
error: { message, code, data },
ip,
userAgent,
},
});
}); });
}); });
}); });

View file

@ -2,8 +2,10 @@ import { BaseLogPayload, LogPayload, LogPayloads, LogResult, LogType } from '@lo
import deepmerge from 'deepmerge'; import deepmerge from 'deepmerge';
import { MiddlewareType } from 'koa'; import { MiddlewareType } from 'koa';
import { IRouterParamContext } from 'koa-router'; import { IRouterParamContext } from 'koa-router';
import pick from 'lodash.pick';
import { nanoid } from 'nanoid'; import { nanoid } from 'nanoid';
import RequestError from '@/errors/RequestError';
import { insertLog } from '@/queries/log'; import { insertLog } from '@/queries/log';
type MergeLog = <T extends LogType>(type: T, payload: LogPayloads[T]) => void; type MergeLog = <T extends LogType>(type: T, payload: LogPayloads[T]) => void;
@ -90,7 +92,13 @@ export default function koaLog<
try { try {
await next(); await next();
} catch (error: unknown) { } catch (error: unknown) {
logger.set({ result: LogResult.Error, error: String(error) }); logger.set({
result: LogResult.Error,
error:
error instanceof RequestError
? pick(error, 'message', 'code', 'data')
: { message: String(error) },
});
throw error; throw error;
} finally { } finally {
await logger.save(); await logger.save();

View file

@ -7,7 +7,7 @@ export enum LogResult {
export interface BaseLogPayload { export interface BaseLogPayload {
result?: LogResult; result?: LogResult;
error?: string; error?: Record<string, unknown>;
ip?: string; ip?: string;
userAgent?: string; userAgent?: string;
applicationId?: string; applicationId?: string;