test: add auth flow integration tests
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
48
src/__tests__/auth.test.ts
Normal file
48
src/__tests__/auth.test.ts
Normal file
@@ -0,0 +1,48 @@
|
||||
import { describe, it, expect, beforeAll, afterAll } from 'vitest';
|
||||
import { buildApp, extractCookie } from './helpers';
|
||||
|
||||
let app: Awaited<ReturnType<typeof buildApp>>;
|
||||
|
||||
beforeAll(async () => { app = await buildApp(); });
|
||||
afterAll(async () => { await app.close(); });
|
||||
|
||||
describe('POST /api/admin/login', () => {
|
||||
it('returns 401 for invalid credentials', async () => {
|
||||
const res = await app.inject({
|
||||
method: 'POST',
|
||||
url: '/api/admin/login',
|
||||
payload: { username: 'nonexistent', password: 'wrong' },
|
||||
});
|
||||
expect(res.statusCode).toBe(401);
|
||||
expect(res.json().success).toBe(false);
|
||||
});
|
||||
|
||||
it('returns 400 for missing fields', async () => {
|
||||
const res = await app.inject({
|
||||
method: 'POST',
|
||||
url: '/api/admin/login',
|
||||
payload: {},
|
||||
});
|
||||
expect(res.statusCode).toBe(400);
|
||||
});
|
||||
});
|
||||
|
||||
describe('POST /api/admin/refresh', () => {
|
||||
it('returns 401 without refresh token', async () => {
|
||||
const res = await app.inject({
|
||||
method: 'POST',
|
||||
url: '/api/admin/refresh',
|
||||
});
|
||||
expect(res.statusCode).toBe(401);
|
||||
});
|
||||
});
|
||||
|
||||
describe('POST /api/admin/logout', () => {
|
||||
it('clears refresh token cookie', async () => {
|
||||
const res = await app.inject({
|
||||
method: 'POST',
|
||||
url: '/api/admin/logout',
|
||||
});
|
||||
expect(res.statusCode).toBeLessThan(500);
|
||||
});
|
||||
});
|
||||
26
src/__tests__/helpers.ts
Normal file
26
src/__tests__/helpers.ts
Normal file
@@ -0,0 +1,26 @@
|
||||
import Fastify from 'fastify';
|
||||
import cookie from '@fastify/cookie';
|
||||
import rateLimit from '@fastify/rate-limit';
|
||||
import authRoutes from '../routes/admin/auth';
|
||||
import totpRoutes from '../routes/admin/totp';
|
||||
|
||||
export async function buildApp() {
|
||||
const app = Fastify({ logger: false });
|
||||
await app.register(cookie);
|
||||
await app.register(rateLimit, { max: 1000, timeWindow: '1 minute' });
|
||||
await app.register(authRoutes, { prefix: '/api/admin' });
|
||||
await app.register(totpRoutes, { prefix: '/api/admin/totp' });
|
||||
return app;
|
||||
}
|
||||
|
||||
export function extractCookie(response: any, name: string): string | undefined {
|
||||
const cookies = response.headers['set-cookie'];
|
||||
if (!cookies) return undefined;
|
||||
const arr = Array.isArray(cookies) ? cookies : [cookies];
|
||||
for (const c of arr) {
|
||||
if (c.startsWith(`${name}=`)) {
|
||||
return c.split(';')[0].split('=')[1];
|
||||
}
|
||||
}
|
||||
return undefined;
|
||||
}
|
||||
Reference in New Issue
Block a user