mirror of
https://github.com/logto-io/logto.git
synced 2025-03-31 22:51:25 -05:00
refactor: minor improvements
This commit is contained in:
parent
93f4ae10ec
commit
2496b57fef
19 changed files with 97 additions and 50 deletions
|
@ -1,7 +1,7 @@
|
|||
import type { LogKey } from '@logto/schemas';
|
||||
import { LogResult, token } from '@logto/schemas';
|
||||
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
import { stringifyError } from '#src/utils/format.js';
|
||||
import { createContextWithRouteParameters } from '#src/utils/test-utils.js';
|
||||
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
import type { LogKey } from '@logto/schemas';
|
||||
import type { PromptDetail } from 'oidc-provider';
|
||||
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
import { createContextWithRouteParameters } from '#src/utils/test-utils.js';
|
||||
|
||||
import { interactionEndedListener, interactionStartedListener } from './interaction.js';
|
||||
|
|
|
@ -22,7 +22,6 @@ mockEsm('nanoid', () => ({
|
|||
|
||||
const koaLog = await pickDefault(import('./koa-audit-log.js'));
|
||||
|
||||
// TODO: test with multiple logs
|
||||
describe('koaAuditLog middleware', () => {
|
||||
const logKey: LogKey = 'Interaction.SignIn.Identifier.VerificationCode.Submit';
|
||||
const mockPayload: LogPayload = {
|
||||
|
@ -67,6 +66,46 @@ describe('koaAuditLog middleware', () => {
|
|||
});
|
||||
});
|
||||
|
||||
it('should insert multiple success logs when needed', async () => {
|
||||
// @ts-expect-error for testing
|
||||
const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = {
|
||||
...createContextWithRouteParameters({ headers: { 'user-agent': userAgent } }),
|
||||
};
|
||||
ctx.request.ip = ip;
|
||||
const additionalMockPayload: LogPayload = { foo: 'bar' };
|
||||
|
||||
const next = async () => {
|
||||
const log = ctx.createLog(logKey);
|
||||
log.append(mockPayload);
|
||||
log.append(additionalMockPayload);
|
||||
const log2 = ctx.createLog(logKey);
|
||||
log2.append(mockPayload);
|
||||
};
|
||||
await koaLog()(ctx, next);
|
||||
|
||||
const basePayload = {
|
||||
...mockPayload,
|
||||
key: logKey,
|
||||
result: LogResult.Success,
|
||||
ip,
|
||||
userAgent,
|
||||
};
|
||||
|
||||
expect(insertLog).toHaveBeenCalledWith({
|
||||
id: nanoIdMock,
|
||||
type: logKey,
|
||||
payload: basePayload,
|
||||
});
|
||||
expect(insertLog).toHaveBeenCalledWith({
|
||||
id: nanoIdMock,
|
||||
type: logKey,
|
||||
payload: {
|
||||
...basePayload,
|
||||
...additionalMockPayload,
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
it('should not log when there is no log type', async () => {
|
||||
// @ts-expect-error for testing
|
||||
const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = {
|
||||
|
@ -112,7 +151,7 @@ describe('koaAuditLog middleware', () => {
|
|||
});
|
||||
});
|
||||
|
||||
it('should insert an error log with the error body when next() throws a RequestError', async () => {
|
||||
it('should update all logs with error result when next() throws a RequestError', async () => {
|
||||
// @ts-expect-error for testing
|
||||
const ctx: WithLogContext<ReturnType<typeof createContextWithRouteParameters>> = {
|
||||
...createContextWithRouteParameters({ headers: { 'user-agent': userAgent } }),
|
||||
|
@ -128,10 +167,13 @@ describe('koaAuditLog middleware', () => {
|
|||
const next = async () => {
|
||||
const log = ctx.createLog(logKey);
|
||||
log.append(mockPayload);
|
||||
const log2 = ctx.createLog(logKey);
|
||||
log2.append(mockPayload);
|
||||
throw error;
|
||||
};
|
||||
await expect(koaLog()(ctx, next)).rejects.toMatchError(error);
|
||||
|
||||
expect(insertLog).toHaveBeenCalledTimes(2);
|
||||
expect(insertLog).toBeCalledWith({
|
||||
id: nanoIdMock,
|
||||
type: logKey,
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
import { Event } from '@logto/schemas';
|
||||
import { mockEsm, pickDefault } from '@logto/shared/esm';
|
||||
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
import { createMockProvider } from '#src/test-utils/oidc-provider.js';
|
||||
import { createContextWithRouteParameters } from '#src/utils/test-utils.js';
|
||||
|
||||
|
|
|
@ -6,7 +6,7 @@ import { mockEsm, mockEsmDefault, mockEsmWithActual, pickDefault } from '@logto/
|
|||
import { mockSignInExperience } from '#src/__mocks__/sign-in-experience.js';
|
||||
import RequestError from '#src/errors/RequestError/index.js';
|
||||
import type koaAuditLog from '#src/middleware/koa-audit-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
import { createMockProvider } from '#src/test-utils/oidc-provider.js';
|
||||
import { createRequester } from '#src/utils/test-utils.js';
|
||||
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
import { PasscodeType, Event } from '@logto/schemas';
|
||||
import { mockEsmWithActual } from '@logto/shared/esm';
|
||||
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
|
||||
import type { SendPasscodePayload } from '../types/index.js';
|
||||
|
||||
|
|
|
@ -43,7 +43,9 @@ export const verifyIdentifierByPasscode = async (
|
|||
const { event, passcode, ...identifier } = payload;
|
||||
const passcodeType = getPasscodeTypeByEvent(event);
|
||||
|
||||
// TODO: @simeng append more log content?
|
||||
// TODO: @Simeng maybe we should just log all interaction payload in every request?
|
||||
const log = createLog(`Interaction.${event}.Identifier.VerificationCode.Submit`);
|
||||
log.append(identifier);
|
||||
|
||||
await verifyPasscode(jti, passcodeType, passcode, identifier);
|
||||
};
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
import { ConnectorType } from '@logto/connector-kit';
|
||||
import { mockEsm } from '@logto/shared/esm';
|
||||
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
|
||||
const { jest } = import.meta;
|
||||
|
||||
|
|
|
@ -2,7 +2,7 @@ import { Event } from '@logto/schemas';
|
|||
import { mockEsm, mockEsmDefault, mockEsmWithActual, pickDefault } from '@logto/shared/esm';
|
||||
|
||||
import RequestError from '#src/errors/RequestError/index.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
import { createMockProvider } from '#src/test-utils/oidc-provider.js';
|
||||
import { createContextWithRouteParameters } from '#src/utils/test-utils.js';
|
||||
|
||||
|
|
|
@ -2,7 +2,7 @@ import { Event } from '@logto/schemas';
|
|||
import { mockEsm, mockEsmWithActual, pickDefault } from '@logto/shared/esm';
|
||||
|
||||
import RequestError from '#src/errors/RequestError/index.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
import { createMockProvider } from '#src/test-utils/oidc-provider.js';
|
||||
import { createContextWithRouteParameters } from '#src/utils/test-utils.js';
|
||||
|
||||
|
|
|
@ -2,7 +2,7 @@ import { Event } from '@logto/schemas';
|
|||
import { mockEsm, mockEsmWithActual, pickDefault } from '@logto/shared/esm';
|
||||
|
||||
import RequestError from '#src/errors/RequestError/index.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
import { createMockProvider } from '#src/test-utils/oidc-provider.js';
|
||||
import { createContextWithRouteParameters } from '#src/utils/test-utils.js';
|
||||
|
||||
|
|
|
@ -2,7 +2,7 @@ import { Event } from '@logto/schemas';
|
|||
import { mockEsm, mockEsmWithActual, pickDefault } from '@logto/shared/esm';
|
||||
|
||||
import RequestError from '#src/errors/RequestError/index.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
import { createMockProvider } from '#src/test-utils/oidc-provider.js';
|
||||
import { createContextWithRouteParameters } from '#src/utils/test-utils.js';
|
||||
|
||||
|
|
|
@ -2,7 +2,7 @@ import { Event } from '@logto/schemas';
|
|||
import { mockEsm, mockEsmWithActual, pickDefault } from '@logto/shared/esm';
|
||||
|
||||
import RequestError from '#src/errors/RequestError/index.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
import { createMockProvider } from '#src/test-utils/oidc-provider.js';
|
||||
import { createContextWithRouteParameters } from '#src/utils/test-utils.js';
|
||||
|
||||
|
|
|
@ -2,7 +2,7 @@ import { Event } from '@logto/schemas';
|
|||
import { mockEsm, mockEsmDefault, mockEsmWithActual, pickDefault } from '@logto/shared/esm';
|
||||
|
||||
import RequestError from '#src/errors/RequestError/index.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-log.js';
|
||||
import { createMockLogContext } from '#src/test-utils/koa-audit-log.js';
|
||||
import { createMockProvider } from '#src/test-utils/oidc-provider.js';
|
||||
import { createContextWithRouteParameters } from '#src/utils/test-utils.js';
|
||||
|
||||
|
|
|
@ -12,10 +12,9 @@
|
|||
"scripts": {
|
||||
"build": "rm -rf lib/ && tsc -p tsconfig.test.json --sourcemap",
|
||||
"test:only": "NODE_OPTIONS=--experimental-vm-modules jest",
|
||||
"test": "pnpm build && pnpm test:api && pnpm test:ui && pnpm test:interaction",
|
||||
"test": "pnpm build && pnpm test:api && pnpm test:ui",
|
||||
"test:api": "pnpm test:only -i ./lib/tests/api",
|
||||
"test:ui": "pnpm test:only -i --config=jest.config.ui.js ./lib/tests/ui",
|
||||
"test:interaction": "pnpm test:only -i ./lib/tests/interaction",
|
||||
"lint": "eslint --ext .ts src",
|
||||
"lint:report": "pnpm lint --format json --output-file report.json",
|
||||
"start": "pnpm test"
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
import type { LogtoConfig } from '@logto/node';
|
||||
import LogtoClient from '@logto/node';
|
||||
import { demoAppApplicationId } from '@logto/schemas/lib/seeds/index.js';
|
||||
import type { Optional } from '@silverhand/essentials';
|
||||
import { assert } from '@silverhand/essentials';
|
||||
import { got } from 'got';
|
||||
|
||||
|
@ -41,6 +42,22 @@ export default class MockClient {
|
|||
return this.rawCookies.join('; ');
|
||||
}
|
||||
|
||||
public get parsedCookies(): Map<string, Optional<string>> {
|
||||
const map = new Map<string, Optional<string>>();
|
||||
|
||||
for (const cookie of this.rawCookies) {
|
||||
for (const element of cookie.split(';')) {
|
||||
const [key, value] = element.trim().split('=');
|
||||
|
||||
if (key) {
|
||||
map.set(key, value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return map;
|
||||
}
|
||||
|
||||
public async initSession(callbackUri = demoAppRedirectUri) {
|
||||
await this.logto.signIn(callbackUri);
|
||||
|
||||
|
|
|
@ -1,36 +1,27 @@
|
|||
import { interaction } from '@logto/schemas';
|
||||
import type { Optional } from '@silverhand/essentials';
|
||||
import { Event, interaction, SignInIdentifier } from '@logto/schemas';
|
||||
import { assert } from '@silverhand/essentials';
|
||||
|
||||
import { deleteUser } from '#src/api/admin-user.js';
|
||||
import { putInteraction } from '#src/api/interaction.js';
|
||||
import { getLogs } from '#src/api/logs.js';
|
||||
import { registerUserWithUsernameAndPassword } from '#src/api/session.js';
|
||||
import MockClient from '#src/client/index.js';
|
||||
import { generatePassword, generateUsername } from '#src/utils.js';
|
||||
|
||||
const parseCookies = (cookies: string[]): Map<string, Optional<string>> => {
|
||||
const map = new Map<string, Optional<string>>();
|
||||
import { enableAllPasswordSignInMethods } from '../interaction/utils/sign-in-experience.js';
|
||||
import { generateNewUserProfile } from '../interaction/utils/user.js';
|
||||
|
||||
for (const cookie of cookies) {
|
||||
for (const element of cookie.split(';')) {
|
||||
const [key, value] = element.trim().split('=');
|
||||
|
||||
if (key) {
|
||||
map.set(key, value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return map;
|
||||
};
|
||||
|
||||
// TODO: @Gao Use new interaction APIs
|
||||
describe('audit logs for interaction', () => {
|
||||
beforeAll(async () => {
|
||||
await enableAllPasswordSignInMethods({
|
||||
identifiers: [SignInIdentifier.Username],
|
||||
password: true,
|
||||
verify: false,
|
||||
});
|
||||
});
|
||||
|
||||
it('should insert log after interaction started and ended', async () => {
|
||||
const client = new MockClient();
|
||||
await client.initSession();
|
||||
const cookies = parseCookies(client.rawCookies);
|
||||
const interactionId = cookies.get('_interaction');
|
||||
const interactionId = client.parsedCookies.get('_interaction');
|
||||
|
||||
assert(interactionId, new Error('No interaction found in cookie'));
|
||||
console.debug('Testing interaction', interactionId);
|
||||
|
@ -42,11 +33,12 @@ describe('audit logs for interaction', () => {
|
|||
expect(createLogs.some((value) => value.payload.interactionId === interactionId)).toBeTruthy();
|
||||
|
||||
// Process interaction with minimum effort
|
||||
const username = generateUsername();
|
||||
const password = generatePassword();
|
||||
const response = await registerUserWithUsernameAndPassword(
|
||||
username,
|
||||
password,
|
||||
const { username, password } = generateNewUserProfile({ username: true, password: true });
|
||||
const response = await putInteraction(
|
||||
{
|
||||
event: Event.Register,
|
||||
profile: { username, password },
|
||||
},
|
||||
client.interactionCookie
|
||||
);
|
||||
await client.processSession(response.redirectTo);
|
||||
|
|
|
@ -31,12 +31,7 @@ export enum Action {
|
|||
|
||||
/**
|
||||
* The union type of all available log keys for interaction.
|
||||
*
|
||||
* The key MUST describe an {@link Action}:
|
||||
*
|
||||
* - {@link Action.Create} (Create a new entity);
|
||||
* - {@link Action.Update} (Update an existing entity);
|
||||
* - {@link Action.Submit} (Submit updated info to an entity, or submit to the system).
|
||||
* The key MUST describe an {@link Action}.
|
||||
*
|
||||
* ### Keys breakdown
|
||||
*
|
||||
|
|
Loading…
Add table
Reference in a new issue