aff52e5672
- 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>
76 lines
2.8 KiB
TypeScript
76 lines
2.8 KiB
TypeScript
import { Router } from 'express';
|
|
import { requireAdmin } from '../middleware/auth';
|
|
import {
|
|
ctiNameSchema as nameSchema,
|
|
createTypeSchema,
|
|
createItemSchema,
|
|
} from '../../../shared/schemas/cti';
|
|
import * as ctiService from '../services/ctiService';
|
|
|
|
const router = Router();
|
|
|
|
// ── Categories ───────────────────────────────────────────────────────────────
|
|
|
|
router.get('/categories', async (_req, res) => {
|
|
res.json(await ctiService.listCategories());
|
|
});
|
|
|
|
router.post('/categories', requireAdmin, async (req, res) => {
|
|
const { name } = nameSchema.parse(req.body);
|
|
res.status(201).json(await ctiService.createCategory(name));
|
|
});
|
|
|
|
router.put('/categories/:id', requireAdmin, async (req, res) => {
|
|
const { name } = nameSchema.parse(req.body);
|
|
res.json(await ctiService.updateCategory(req.params.id, name));
|
|
});
|
|
|
|
router.delete('/categories/:id', requireAdmin, async (req, res) => {
|
|
await ctiService.deleteCategory(req.params.id);
|
|
res.status(204).send();
|
|
});
|
|
|
|
// ── Types ────────────────────────────────────────────────────────────────────
|
|
|
|
router.get('/types', async (req, res) => {
|
|
res.json(await ctiService.listTypes(req.query.categoryId as string | undefined));
|
|
});
|
|
|
|
router.post('/types', requireAdmin, async (req, res) => {
|
|
const { name, categoryId } = createTypeSchema.parse(req.body);
|
|
res.status(201).json(await ctiService.createType(name, categoryId));
|
|
});
|
|
|
|
router.put('/types/:id', requireAdmin, async (req, res) => {
|
|
const { name } = nameSchema.parse(req.body);
|
|
res.json(await ctiService.updateType(req.params.id, name));
|
|
});
|
|
|
|
router.delete('/types/:id', requireAdmin, async (req, res) => {
|
|
await ctiService.deleteType(req.params.id);
|
|
res.status(204).send();
|
|
});
|
|
|
|
// ── Items ────────────────────────────────────────────────────────────────────
|
|
|
|
router.get('/items', async (req, res) => {
|
|
res.json(await ctiService.listItems(req.query.typeId as string | undefined));
|
|
});
|
|
|
|
router.post('/items', requireAdmin, async (req, res) => {
|
|
const { name, typeId } = createItemSchema.parse(req.body);
|
|
res.status(201).json(await ctiService.createItem(name, typeId));
|
|
});
|
|
|
|
router.put('/items/:id', requireAdmin, async (req, res) => {
|
|
const { name } = nameSchema.parse(req.body);
|
|
res.json(await ctiService.updateItem(req.params.id, name));
|
|
});
|
|
|
|
router.delete('/items/:id', requireAdmin, async (req, res) => {
|
|
await ctiService.deleteItem(req.params.id);
|
|
res.status(204).send();
|
|
});
|
|
|
|
export default router;
|