0
Fork 0
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:
Gao Sun 2022-12-18 12:11:00 +08:00
parent 93f4ae10ec
commit 2496b57fef
No known key found for this signature in database
GPG key ID: 13EBE123E4773688
19 changed files with 97 additions and 50 deletions

View file

@ -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';

View file

@ -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';

View file

@ -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,

View file

@ -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';

View file

@ -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';

View file

@ -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';

View file

@ -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);
};

View file

@ -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;

View file

@ -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';

View file

@ -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';

View file

@ -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';

View file

@ -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';

View file

@ -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';

View file

@ -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';

View file

@ -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"

View file

@ -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);

View file

@ -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);

View file

@ -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
*