Files
ReadMeABook/tests/api/auth-change-password.routes.test.ts
T
kikootwo 94dbaf073b Add backend unit test framework and modularize settings UI
Introduced a Vitest-based backend unit testing framework with supporting scripts, helpers, and GitHub Actions integration. Refactored the admin settings page to a modular architecture, splitting monolithic logic into feature-specific tabs and hooks for improved maintainability and testability. Updated documentation to reflect the new testing setup and settings architecture, and added new dependencies for testing utilities.
2026-01-28 11:41:59 -05:00

162 lines
4.8 KiB
TypeScript

/**
* Component: Change Password API Route Tests
* Documentation: documentation/backend/services/auth.md
*/
import { beforeEach, describe, expect, it, vi } from 'vitest';
import { createPrismaMock } from '../helpers/prisma';
const prismaMock = createPrismaMock();
const bcryptMock = {
compare: vi.fn(),
hash: vi.fn(),
};
const encryptionMock = {
decrypt: vi.fn(),
encrypt: vi.fn(),
};
const requireAuthMock = vi.fn();
vi.mock('@/lib/db', () => ({
prisma: prismaMock,
}));
vi.mock('bcrypt', () => ({
default: bcryptMock,
...bcryptMock,
}));
vi.mock('@/lib/services/encryption.service', () => ({
getEncryptionService: () => encryptionMock,
}));
vi.mock('@/lib/middleware/auth', () => ({
requireAuth: requireAuthMock,
}));
const makeRequest = (body: Record<string, any>) => ({
json: vi.fn().mockResolvedValue(body),
});
describe('Change password route', () => {
beforeEach(() => {
vi.clearAllMocks();
requireAuthMock.mockImplementation((_req: any, handler: any) =>
handler({ user: { id: 'user-1' } })
);
});
it('validates required fields', async () => {
const { POST } = await import('@/app/api/auth/change-password/route');
const response = await POST(makeRequest({ currentPassword: 'old' }) as any);
const payload = await response.json();
expect(response.status).toBe(400);
expect(payload.error).toMatch(/required/i);
});
it('rejects short passwords', async () => {
const { POST } = await import('@/app/api/auth/change-password/route');
const response = await POST(
makeRequest({ currentPassword: 'old', newPassword: 'short', confirmPassword: 'short' }) as any
);
const payload = await response.json();
expect(response.status).toBe(400);
expect(payload.error).toMatch(/at least 8 characters/i);
});
it('blocks non-local users', async () => {
prismaMock.user.findUnique.mockResolvedValue({
id: 'user-1',
authProvider: 'plex',
authToken: 'enc-hash',
plexId: 'plex-1',
plexUsername: 'user',
});
const { POST } = await import('@/app/api/auth/change-password/route');
const response = await POST(
makeRequest({ currentPassword: 'oldpass', newPassword: 'newpass123', confirmPassword: 'newpass123' }) as any
);
const payload = await response.json();
expect(response.status).toBe(403);
expect(payload.error).toMatch(/local users/i);
});
it('rejects incorrect current password', async () => {
prismaMock.user.findUnique.mockResolvedValue({
id: 'user-1',
authProvider: 'local',
authToken: 'enc-hash',
plexId: 'local-user',
plexUsername: 'user',
});
encryptionMock.decrypt.mockReturnValue('hash');
bcryptMock.compare.mockResolvedValue(false);
const { POST } = await import('@/app/api/auth/change-password/route');
const response = await POST(
makeRequest({ currentPassword: 'wrong', newPassword: 'newpass123', confirmPassword: 'newpass123' }) as any
);
const payload = await response.json();
expect(response.status).toBe(400);
expect(payload.error).toMatch(/incorrect/i);
});
it('returns error when decryption fails', async () => {
prismaMock.user.findUnique.mockResolvedValue({
id: 'user-1',
authProvider: 'local',
authToken: 'enc-hash',
plexId: 'local-user',
plexUsername: 'user',
});
encryptionMock.decrypt.mockImplementation(() => {
throw new Error('decrypt failed');
});
const { POST } = await import('@/app/api/auth/change-password/route');
const response = await POST(
makeRequest({ currentPassword: 'oldpass', newPassword: 'newpass123', confirmPassword: 'newpass123' }) as any
);
const payload = await response.json();
expect(response.status).toBe(500);
expect(payload.error).toMatch(/verify current password/i);
});
it('updates password for local user', async () => {
prismaMock.user.findUnique.mockResolvedValue({
id: 'user-1',
authProvider: 'local',
authToken: 'enc-hash',
plexId: 'local-user',
plexUsername: 'user',
});
encryptionMock.decrypt.mockReturnValue('hash');
bcryptMock.compare.mockResolvedValue(true);
bcryptMock.hash.mockResolvedValue('new-hash');
encryptionMock.encrypt.mockReturnValue('enc-new-hash');
prismaMock.user.update.mockResolvedValue({});
const { POST } = await import('@/app/api/auth/change-password/route');
const response = await POST(
makeRequest({ currentPassword: 'oldpass', newPassword: 'newpass123', confirmPassword: 'newpass123' }) as any
);
const payload = await response.json();
expect(response.status).toBe(200);
expect(payload.success).toBe(true);
expect(prismaMock.user.update).toHaveBeenCalledWith(
expect.objectContaining({
data: expect.objectContaining({ authToken: 'enc-new-hash' }),
})
);
});
});