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

feat(core,schemas): make it type-safer to log (#656)

This commit is contained in:
IceHe.xyz 2022-04-26 11:31:07 +08:00 committed by GitHub
parent 68a64d3ef5
commit 3aa4342f2e
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 147 additions and 97 deletions

View file

@ -25,6 +25,7 @@
"@silverhand/essentials": "^1.1.0", "@silverhand/essentials": "^1.1.0",
"dayjs": "^1.10.5", "dayjs": "^1.10.5",
"decamelize": "^5.0.0", "decamelize": "^5.0.0",
"deepmerge": "^4.2.2",
"dotenv": "^16.0.0", "dotenv": "^16.0.0",
"got": "^11.8.2", "got": "^11.8.2",
"i18next": "^21.0.0", "i18next": "^21.0.0",

View file

@ -1,12 +1,14 @@
import { LogType, LogResult } from '@logto/schemas'; import { LogPayload, LogResult } from '@logto/schemas';
import { insertLog } from '@/queries/log'; import { insertLog } from '@/queries/log';
import { createContextWithRouteParameters } from '@/utils/test-utils'; import { createContextWithRouteParameters } from '@/utils/test-utils';
import koaLog, { WithLogContext, LogContext } from './koa-log'; import koaLog, { WithLogContext } from './koa-log';
const nanoIdMock = 'mockId'; const nanoIdMock = 'mockId';
const log = jest.fn();
jest.mock('@/queries/log', () => ({ jest.mock('@/queries/log', () => ({
insertLog: jest.fn(async () => Promise.resolve()), insertLog: jest.fn(async () => Promise.resolve()),
})); }));
@ -19,11 +21,10 @@ describe('koaLog middleware', () => {
const insertLogMock = insertLog as jest.Mock; const insertLogMock = insertLog as jest.Mock;
const next = jest.fn(); const next = jest.fn();
const logMock: Partial<LogContext> = { const type = 'SignInUsernamePassword';
type: LogType.SignInUsernamePassword, const payload: LogPayload = {
applicationId: 'foo',
userId: 'foo', userId: 'foo',
username: 'Foo Bar', username: 'Bar',
}; };
afterEach(() => { afterEach(() => {
@ -33,21 +34,20 @@ describe('koaLog middleware', () => {
it('insert log with success response', async () => { it('insert log with success response', async () => {
const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = { const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = {
...createContextWithRouteParameters(), ...createContextWithRouteParameters(),
log: {}, // Bypass middleware context type assert log, // Bypass middleware context type assert
}; };
next.mockImplementationOnce(async () => { next.mockImplementationOnce(async () => {
ctx.log = logMock; ctx.log(type, payload);
}); });
await koaLog()(ctx, next); await koaLog()(ctx, next);
const { type, ...rest } = logMock;
expect(insertLogMock).toBeCalledWith({ expect(insertLogMock).toBeCalledWith({
id: nanoIdMock, id: nanoIdMock,
type, type,
payload: { payload: {
...rest, ...payload,
result: LogResult.Success, result: LogResult.Success,
}, },
}); });
@ -56,7 +56,7 @@ describe('koaLog middleware', () => {
it('should not block request if insertLog throws error', async () => { it('should not block request if insertLog throws error', async () => {
const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = { const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = {
...createContextWithRouteParameters(), ...createContextWithRouteParameters(),
log: {}, // Bypass middleware context type assert log, // Bypass middleware context type assert
}; };
const error = new Error('Failed to insert log'); const error = new Error('Failed to insert log');
@ -65,17 +65,16 @@ describe('koaLog middleware', () => {
}); });
next.mockImplementationOnce(async () => { next.mockImplementationOnce(async () => {
ctx.log = logMock; ctx.log(type, payload);
}); });
await koaLog()(ctx, next); await koaLog()(ctx, next);
const { type, ...rest } = logMock;
expect(insertLogMock).toBeCalledWith({ expect(insertLogMock).toBeCalledWith({
id: nanoIdMock, id: nanoIdMock,
type, type,
payload: { payload: {
...rest, ...payload,
result: LogResult.Success, result: LogResult.Success,
}, },
}); });
@ -84,24 +83,23 @@ describe('koaLog middleware', () => {
it('should insert log with failed result if next throws error', async () => { it('should insert log with failed result if next throws error', async () => {
const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = { const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = {
...createContextWithRouteParameters(), ...createContextWithRouteParameters(),
log: {}, // Bypass middleware context type assert log, // Bypass middleware context type assert
}; };
const error = new Error('next error'); const error = new Error('next error');
next.mockImplementationOnce(async () => { next.mockImplementationOnce(async () => {
ctx.log = logMock; ctx.log(type, payload);
throw error; throw error;
}); });
await expect(koaLog()(ctx, next)).rejects.toMatchError(error); await expect(koaLog()(ctx, next)).rejects.toMatchError(error);
const { type, ...rest } = logMock;
expect(insertLogMock).toBeCalledWith({ expect(insertLogMock).toBeCalledWith({
id: nanoIdMock, id: nanoIdMock,
type, type,
payload: { payload: {
...rest, ...payload,
result: LogResult.Error, result: LogResult.Error,
error: String(error), error: String(error),
}, },

View file

@ -1,33 +1,21 @@
import { LogResult, LogType } from '@logto/schemas'; import { LogPayload, LogPayloads, LogResult, LogType } from '@logto/schemas';
import { Context, MiddlewareType } from 'koa'; import { Optional } from '@silverhand/essentials';
import deepmerge from 'deepmerge';
import { MiddlewareType } from 'koa';
import { nanoid } from 'nanoid'; import { nanoid } from 'nanoid';
import { insertLog } from '@/queries/log'; import { insertLog } from '@/queries/log';
export type WithLogContext<ContextT> = ContextT & { export type WithLogContext<ContextT> = ContextT & {
log: LogContext; log: <T extends LogType>(type: T, payload: LogPayloads[T]) => void;
}; };
export interface LogContext { const saveLog = async (type: LogType, payload: LogPayload) => {
[key: string]: unknown;
type?: LogType;
}
const log = async (ctx: WithLogContext<Context>, result: LogResult) => {
const { type, ...rest } = ctx.log;
if (!type) {
return;
}
try { try {
await insertLog({ await insertLog({
id: nanoid(), id: nanoid(),
type, type,
payload: { payload,
...rest,
result,
},
}); });
} catch (error: unknown) { } catch (error: unknown) {
console.error('An error occurred while inserting log'); console.error('An error occurred while inserting log');
@ -41,14 +29,33 @@ export default function koaLog<StateT, ContextT, ResponseBodyT>(): MiddlewareTyp
ResponseBodyT ResponseBodyT
> { > {
return async (ctx, next) => { return async (ctx, next) => {
ctx.log = {}; // eslint-disable-next-line @silverhand/fp/no-let
let logType: Optional<LogType>;
// eslint-disable-next-line @silverhand/fp/no-let
let logPayload: LogPayload = {};
ctx.log = (type, payload) => {
if (logType !== type) {
// eslint-disable-next-line @silverhand/fp/no-mutation
logPayload = {}; // Reset payload when type changes
}
// eslint-disable-next-line @silverhand/fp/no-mutation
logType = type; // Use first initialized log type
// eslint-disable-next-line @silverhand/fp/no-mutation
logPayload = deepmerge(logPayload, payload);
};
try { try {
await next(); await next();
await log(ctx, LogResult.Success);
if (logType) {
await saveLog(logType, { ...logPayload, result: LogResult.Success });
}
} catch (error: unknown) { } catch (error: unknown) {
ctx.log.error = String(error); if (logType) {
await log(ctx, LogResult.Error); await saveLog(logType, { ...logPayload, result: LogResult.Error, error: String(error) });
}
throw error; throw error;
} }
}; };

View file

@ -153,7 +153,7 @@ describe('sessionRoutes', () => {
provider: new Provider(''), provider: new Provider(''),
middlewares: [ middlewares: [
async (ctx, next) => { async (ctx, next) => {
ctx.log = {}; ctx.log = jest.fn();
return next(); return next();
}, },

View file

@ -2,7 +2,7 @@
import path from 'path'; import path from 'path';
import { LogtoErrorCode } from '@logto/phrases'; import { LogtoErrorCode } from '@logto/phrases';
import { LogType, PasscodeType, userInfoSelectFields } from '@logto/schemas'; import { PasscodeType, userInfoSelectFields } from '@logto/schemas';
import { conditional } from '@silverhand/essentials'; import { conditional } from '@silverhand/essentials';
import pick from 'lodash.pick'; import pick from 'lodash.pick';
import { Provider } from 'oidc-provider'; import { Provider } from 'oidc-provider';
@ -67,13 +67,14 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
}), }),
}), }),
async (ctx, next) => { async (ctx, next) => {
const { jti } = await provider.interactionDetails(ctx.req, ctx.res);
const { username, password } = ctx.guard.body; const { username, password } = ctx.guard.body;
ctx.log.type = LogType.SignInUsernamePassword; const type = 'SignInUsernamePassword';
ctx.log.username = username; ctx.log(type, { sessionId: jti, username });
assertThat(password, 'session.insufficient_info'); assertThat(password, 'session.insufficient_info');
const { id } = await findUserByUsernameAndPassword(username, password); const { id } = await findUserByUsernameAndPassword(username, password);
ctx.log.userId = id; ctx.log(type, { userId: id });
await assignInteractionResults(ctx, provider, { login: { accountId: id } }); await assignInteractionResults(ctx, provider, { login: { accountId: id } });
return next(); return next();
@ -84,12 +85,10 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
'/session/sign-in/passwordless/sms/send-passcode', '/session/sign-in/passwordless/sms/send-passcode',
koaGuard({ body: object({ phone: string().regex(phoneRegEx) }) }), koaGuard({ body: object({ phone: string().regex(phoneRegEx) }) }),
async (ctx, next) => { async (ctx, next) => {
const { phone } = ctx.guard.body;
ctx.log.type = LogType.SignInSmsSendPasscode;
ctx.log.phone = phone;
const { jti } = await provider.interactionDetails(ctx.req, ctx.res); const { jti } = await provider.interactionDetails(ctx.req, ctx.res);
ctx.log.sessionId = jti; const { phone } = ctx.guard.body;
const type = 'SignInSmsSendPasscode';
ctx.log(type, { sessionId: jti, phone });
assertThat( assertThat(
await hasUserWithPhone(phone), await hasUserWithPhone(phone),
@ -97,7 +96,7 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
); );
const passcode = await createPasscode(jti, PasscodeType.SignIn, { phone }); const passcode = await createPasscode(jti, PasscodeType.SignIn, { phone });
ctx.log.passcode = passcode; ctx.log(type, { passcode });
await sendPasscode(passcode); await sendPasscode(passcode);
ctx.status = 204; ctx.status = 204;
@ -110,13 +109,10 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
'/session/sign-in/passwordless/sms/verify-passcode', '/session/sign-in/passwordless/sms/verify-passcode',
koaGuard({ body: object({ phone: string().regex(phoneRegEx), code: string() }) }), koaGuard({ body: object({ phone: string().regex(phoneRegEx), code: string() }) }),
async (ctx, next) => { async (ctx, next) => {
const { phone, code } = ctx.guard.body;
ctx.log.type = LogType.SignInSms;
ctx.log.phone = phone;
ctx.log.passcode = code;
const { jti } = await provider.interactionDetails(ctx.req, ctx.res); const { jti } = await provider.interactionDetails(ctx.req, ctx.res);
ctx.log.sessionId = jti; const { phone, code } = ctx.guard.body;
const type = 'SignInSms';
ctx.log(type, { sessionId: jti, phone, code });
assertThat( assertThat(
await hasUserWithPhone(phone), await hasUserWithPhone(phone),
@ -125,7 +121,7 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
await verifyPasscode(jti, PasscodeType.SignIn, code, { phone }); await verifyPasscode(jti, PasscodeType.SignIn, code, { phone });
const { id } = await findUserByPhone(phone); const { id } = await findUserByPhone(phone);
ctx.log.userId = id; ctx.log(type, { userId: id });
await assignInteractionResults(ctx, provider, { login: { accountId: id } }); await assignInteractionResults(ctx, provider, { login: { accountId: id } });
@ -137,12 +133,10 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
'/session/sign-in/passwordless/email/send-passcode', '/session/sign-in/passwordless/email/send-passcode',
koaGuard({ body: object({ email: string().regex(emailRegEx) }) }), koaGuard({ body: object({ email: string().regex(emailRegEx) }) }),
async (ctx, next) => { async (ctx, next) => {
const { email } = ctx.guard.body;
ctx.log.type = LogType.SignInEmailSendPasscode;
ctx.log.email = email;
const { jti } = await provider.interactionDetails(ctx.req, ctx.res); const { jti } = await provider.interactionDetails(ctx.req, ctx.res);
ctx.log.sessionId = jti; const { email } = ctx.guard.body;
const type = 'SignInEmailSendPasscode';
ctx.log(type, { sessionId: jti, email });
assertThat( assertThat(
await hasUserWithEmail(email), await hasUserWithEmail(email),
@ -150,7 +144,7 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
); );
const passcode = await createPasscode(jti, PasscodeType.SignIn, { email }); const passcode = await createPasscode(jti, PasscodeType.SignIn, { email });
ctx.log.passcode = passcode; ctx.log(type, { passcode });
await sendPasscode(passcode); await sendPasscode(passcode);
ctx.status = 204; ctx.status = 204;
@ -163,13 +157,10 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
'/session/sign-in/passwordless/email/verify-passcode', '/session/sign-in/passwordless/email/verify-passcode',
koaGuard({ body: object({ email: string().regex(emailRegEx), code: string() }) }), koaGuard({ body: object({ email: string().regex(emailRegEx), code: string() }) }),
async (ctx, next) => { async (ctx, next) => {
const { email, code } = ctx.guard.body;
ctx.log.type = LogType.SignInEmail;
ctx.log.email = email;
ctx.log.passcode = code;
const { jti } = await provider.interactionDetails(ctx.req, ctx.res); const { jti } = await provider.interactionDetails(ctx.req, ctx.res);
ctx.log.sessionId = jti; const { email, code } = ctx.guard.body;
const type = 'SignInEmail';
ctx.log(type, { sessionId: jti, email, code });
assertThat( assertThat(
await hasUserWithEmail(email), await hasUserWithEmail(email),
@ -178,7 +169,7 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
await verifyPasscode(jti, PasscodeType.SignIn, code, { email }); await verifyPasscode(jti, PasscodeType.SignIn, code, { email });
const { id } = await findUserByEmail(email); const { id } = await findUserByEmail(email);
ctx.log.userId = id; ctx.log(type, { userId: id });
await assignInteractionResults(ctx, provider, { login: { accountId: id } }); await assignInteractionResults(ctx, provider, { login: { accountId: id } });
@ -198,11 +189,8 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
}), }),
async (ctx, next) => { async (ctx, next) => {
const { connectorId, code, state, redirectUri } = ctx.guard.body; const { connectorId, code, state, redirectUri } = ctx.guard.body;
ctx.log.type = LogType.SignInSocial; const type = 'SignInSocial';
ctx.log.connectorId = connectorId; ctx.log(type, { connectorId, code, state, redirectUri });
ctx.log.code = code;
ctx.log.state = state;
ctx.log.redirectUri = redirectUri;
if (!code) { if (!code) {
assertThat(state && redirectUri, 'session.insufficient_info'); assertThat(state && redirectUri, 'session.insufficient_info');
@ -210,13 +198,13 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
assertThat(connector.connector.enabled, 'connector.not_enabled'); assertThat(connector.connector.enabled, 'connector.not_enabled');
const redirectTo = await connector.getAuthorizationUri(redirectUri, state); const redirectTo = await connector.getAuthorizationUri(redirectUri, state);
ctx.body = { redirectTo }; ctx.body = { redirectTo };
ctx.log.redirectTo = redirectTo; ctx.log(type, { redirectTo });
return next(); return next();
} }
const userInfo = await getUserInfoByAuthCode(connectorId, code, redirectUri); const userInfo = await getUserInfoByAuthCode(connectorId, code, redirectUri);
ctx.log.userInfo = userInfo; ctx.log(type, { userInfo });
if (!(await hasUserWithIdentity(connectorId, userInfo.id))) { if (!(await hasUserWithIdentity(connectorId, userInfo.id))) {
await assignInteractionResults(ctx, provider, { connectorId, userInfo }, true); await assignInteractionResults(ctx, provider, { connectorId, userInfo }, true);
@ -231,7 +219,7 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
} }
const { id, identities } = await findUserByIdentity(connectorId, userInfo.id); const { id, identities } = await findUserByIdentity(connectorId, userInfo.id);
ctx.log.userId = id; ctx.log(type, { userId: id });
// Update social connector's user info // Update social connector's user info
await updateUserById(id, { await updateUserById(id, {
@ -249,21 +237,21 @@ export default function sessionRoutes<T extends AnonymousRouter>(router: T, prov
body: object({ connectorId: string() }), body: object({ connectorId: string() }),
}), }),
async (ctx, next) => { async (ctx, next) => {
const { connectorId } = ctx.guard.body; const { jti, result } = await provider.interactionDetails(ctx.req, ctx.res);
ctx.log.type = LogType.SignInSocialBind;
ctx.log.connectorId = connectorId;
const { result } = await provider.interactionDetails(ctx.req, ctx.res);
assertThat(result, 'session.connector_session_not_found'); assertThat(result, 'session.connector_session_not_found');
const { connectorId } = ctx.guard.body;
const type = 'SignInSocialBind';
ctx.log(type, { sessionId: jti, connectorId });
const userInfo = await getUserInfoFromInteractionResult(connectorId, result); const userInfo = await getUserInfoFromInteractionResult(connectorId, result);
ctx.log.userInfo = userInfo; ctx.log(type, { userInfo });
const relatedInfo = await findSocialRelatedUser(userInfo); const relatedInfo = await findSocialRelatedUser(userInfo);
assertThat(relatedInfo, 'session.connector_session_not_found'); assertThat(relatedInfo, 'session.connector_session_not_found');
const { id, identities } = relatedInfo[1]; const { id, identities } = relatedInfo[1];
ctx.log.userId = id; ctx.log(type, { userId: id });
await updateUserById(id, { await updateUserById(id, {
identities: { ...identities, [connectorId]: { userId: userInfo.id, details: userInfo } }, identities: { ...identities, [connectorId]: { userId: userInfo.id, details: userInfo } },

View file

@ -5,7 +5,9 @@ import { z } from 'zod';
* Commonly Used * Commonly Used
*/ */
export const arbitraryObjectGuard = z.object({}).catchall(z.unknown()); // Cannot declare `z.object({}).catchall(z.unknown().optional())` to guard `{ [key: string]?: unknown }` (invalid type),
// so do it another way to guard `{ [x: string]: unknown; } | {}`.
export const arbitraryObjectGuard = z.union([z.object({}).catchall(z.unknown()), z.object({})]);
export type ArbitraryObject = z.infer<typeof arbitraryObjectGuard>; export type ArbitraryObject = z.infer<typeof arbitraryObjectGuard>;

View file

@ -1,14 +1,66 @@
export enum LogType { import { Passcode } from '../db-entries';
SignInUsernamePassword = 'SignInUsernamePassword',
SignInEmail = 'SignInEmail',
SignInEmailSendPasscode = 'SignInEmailSendPasscode',
SignInSms = 'SignInSms',
SignInSmsSendPasscode = 'SignInSmsSendPasscode',
SignInSocial = 'SignInSocial',
SignInSocialBind = 'SignInSocialBind',
}
export enum LogResult { export enum LogResult {
Success = 'Success', Success = 'Success',
Error = 'Error', Error = 'Error',
} }
interface BaseLogPayload {
sessionId?: string;
result?: LogResult;
error?: string;
}
interface SignInUsernamePasswordLogPayload extends BaseLogPayload {
userId?: string;
username?: string;
}
interface SignInEmailSendPasscodeLogPayload extends BaseLogPayload {
email?: string;
passcode?: Passcode;
}
interface SignInEmailLogPayload extends BaseLogPayload {
email?: string;
code?: string;
userId?: string;
}
interface SignInSmsSendPasscodeLogPayload extends BaseLogPayload {
phone?: string;
passcode?: Passcode;
}
interface SignInSmsLogPayload extends BaseLogPayload {
phone?: string;
code?: string;
userId?: string;
}
interface SignInSocialBindLogPayload extends BaseLogPayload {
connectorId?: string;
userInfo?: object;
userId?: string;
}
interface SignInSocialLogPayload extends SignInSocialBindLogPayload {
code?: string;
state?: string;
redirectUri?: string;
redirectTo?: string;
}
export type LogPayloads = {
SignInUsernamePassword: SignInUsernamePasswordLogPayload;
SignInEmailSendPasscode: SignInEmailSendPasscodeLogPayload;
SignInEmail: SignInEmailLogPayload;
SignInSmsSendPasscode: SignInSmsSendPasscodeLogPayload;
SignInSms: SignInSmsLogPayload;
SignInSocialBind: SignInSocialBindLogPayload;
SignInSocial: SignInSocialLogPayload;
};
export type LogType = keyof LogPayloads;
export type LogPayload = LogPayloads[LogType];

2
pnpm-lock.yaml generated
View file

@ -273,6 +273,7 @@ importers:
copyfiles: ^2.4.1 copyfiles: ^2.4.1
dayjs: ^1.10.5 dayjs: ^1.10.5
decamelize: ^5.0.0 decamelize: ^5.0.0
deepmerge: ^4.2.2
dotenv: ^16.0.0 dotenv: ^16.0.0
eslint: ^8.10.0 eslint: ^8.10.0
got: ^11.8.2 got: ^11.8.2
@ -314,6 +315,7 @@ importers:
'@silverhand/essentials': 1.1.2 '@silverhand/essentials': 1.1.2
dayjs: 1.10.7 dayjs: 1.10.7
decamelize: 5.0.1 decamelize: 5.0.1
deepmerge: 4.2.2
dotenv: 16.0.0 dotenv: 16.0.0
got: 11.8.3 got: 11.8.3
i18next: 21.6.12 i18next: 21.6.12