Phase 1a: shared schemas, service layer, server tooling
- shared/schemas/: move Zod schemas out of routes so client + server share them - shared/types.ts: inferred types and enums for cross-package use - server tsconfig rootDir raised to ".." so shared/ compiles in-tree - server/src/services/: ticket, comment, cti, user, auth, notification (stub), search (stub) - Routes thinned to validate-delegate-return; business logic now testable in isolation - server/src/lib/httpError.ts: typed HttpError replaces ad-hoc throw shapes - server/src/lib/logger.ts: pino structured logging replaces console.log - autoClose job delegates to ticketService.closeStale() - express-rate-limit on /api/auth/login (10 / 15min / IP) - vitest + vitest-mock-extended; 20 service-level tests cover auth, ticket, comment, user flows - CI: lint + test jobs before docker builds Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,83 @@
|
||||
import { describe, it, expect } from 'vitest';
|
||||
import { prismaMock } from '../test/setup';
|
||||
import { addComment, deleteComment } from './commentService';
|
||||
import { HttpError } from '../lib/httpError';
|
||||
|
||||
describe('commentService.addComment', () => {
|
||||
it('404s when ticket not found', async () => {
|
||||
prismaMock.ticket.findFirst.mockResolvedValue(null);
|
||||
await expect(addComment('V1', 'body', 'u1')).rejects.toMatchObject({ status: 404 });
|
||||
});
|
||||
|
||||
it('accepts either id or displayId and writes audit + comment', async () => {
|
||||
prismaMock.ticket.findFirst.mockResolvedValue({
|
||||
id: 'tid',
|
||||
displayId: 'V111',
|
||||
title: 't',
|
||||
overview: 'o',
|
||||
severity: 3,
|
||||
status: 'OPEN',
|
||||
categoryId: 'c',
|
||||
typeId: 'ty',
|
||||
itemId: 'i',
|
||||
assigneeId: null,
|
||||
createdById: 'u1',
|
||||
createdAt: new Date(),
|
||||
updatedAt: new Date(),
|
||||
resolvedAt: null,
|
||||
});
|
||||
prismaMock.comment.create.mockResolvedValue({
|
||||
id: 'cid',
|
||||
body: 'hi',
|
||||
ticketId: 'tid',
|
||||
authorId: 'u1',
|
||||
createdAt: new Date(),
|
||||
});
|
||||
|
||||
await addComment('V111', 'hi', 'u1');
|
||||
expect(prismaMock.comment.create).toHaveBeenCalledWith(
|
||||
expect.objectContaining({ data: expect.objectContaining({ ticketId: 'tid', body: 'hi' }) }),
|
||||
);
|
||||
expect(prismaMock.auditLog.create).toHaveBeenCalledWith(
|
||||
expect.objectContaining({
|
||||
data: expect.objectContaining({ action: 'COMMENT_ADDED', detail: 'hi' }),
|
||||
}),
|
||||
);
|
||||
});
|
||||
});
|
||||
|
||||
describe('commentService.deleteComment', () => {
|
||||
const baseComment = {
|
||||
id: 'cid',
|
||||
body: 'x',
|
||||
ticketId: 'tid',
|
||||
authorId: 'author1',
|
||||
createdAt: new Date(),
|
||||
};
|
||||
|
||||
it('rejects non-author non-admin with 403', async () => {
|
||||
prismaMock.comment.findUnique.mockResolvedValue(baseComment);
|
||||
await expect(
|
||||
deleteComment('cid', { id: 'other', role: 'AGENT' }),
|
||||
).rejects.toMatchObject({ status: 403 });
|
||||
});
|
||||
|
||||
it('allows the author to delete', async () => {
|
||||
prismaMock.comment.findUnique.mockResolvedValue(baseComment);
|
||||
await expect(deleteComment('cid', { id: 'author1', role: 'AGENT' })).resolves.toBeUndefined();
|
||||
expect(prismaMock.comment.delete).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('allows admin to delete', async () => {
|
||||
prismaMock.comment.findUnique.mockResolvedValue(baseComment);
|
||||
await expect(deleteComment('cid', { id: 'other', role: 'ADMIN' })).resolves.toBeUndefined();
|
||||
expect(prismaMock.comment.delete).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('404s when comment missing', async () => {
|
||||
prismaMock.comment.findUnique.mockResolvedValue(null);
|
||||
await expect(
|
||||
deleteComment('missing', { id: 'u', role: 'ADMIN' }),
|
||||
).rejects.toThrow(HttpError);
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user