"use strict"; /** * BitacoraObraController - Controller de bitácora de obra * * Endpoints REST para gestión de bitácora diaria de obra. * * @module Progress */ Object.defineProperty(exports, "__esModule", { value: true }); exports.createBitacoraObraController = createBitacoraObraController; const express_1 = require("express"); const bitacora_obra_service_1 = require("../services/bitacora-obra.service"); const auth_middleware_1 = require("../../auth/middleware/auth.middleware"); const auth_service_1 = require("../../auth/services/auth.service"); const bitacora_obra_entity_1 = require("../entities/bitacora-obra.entity"); const user_entity_1 = require("../../core/entities/user.entity"); const tenant_entity_1 = require("../../core/entities/tenant.entity"); const refresh_token_entity_1 = require("../../auth/entities/refresh-token.entity"); /** * Crear router de bitácora de obra */ function createBitacoraObraController(dataSource) { const router = (0, express_1.Router)(); // Repositorios const bitacoraRepository = dataSource.getRepository(bitacora_obra_entity_1.BitacoraObra); const userRepository = dataSource.getRepository(user_entity_1.User); const tenantRepository = dataSource.getRepository(tenant_entity_1.Tenant); const refreshTokenRepository = dataSource.getRepository(refresh_token_entity_1.RefreshToken); // Servicios const bitacoraService = new bitacora_obra_service_1.BitacoraObraService(bitacoraRepository); const authService = new auth_service_1.AuthService(userRepository, tenantRepository, refreshTokenRepository); const authMiddleware = new auth_middleware_1.AuthMiddleware(authService, dataSource); // Helper para crear contexto de servicio const getContext = (req) => { if (!req.tenantId) { throw new Error('Tenant ID is required'); } return { tenantId: req.tenantId, userId: req.user?.sub, }; }; /** * GET /bitacora * Listar entradas de bitácora por fraccionamiento */ router.get('/', authMiddleware.authenticate, async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const fraccionamientoId = req.query.fraccionamientoId; if (!fraccionamientoId) { res.status(400).json({ error: 'Bad Request', message: 'fraccionamientoId is required' }); return; } const page = parseInt(req.query.page) || 1; const limit = Math.min(parseInt(req.query.limit) || 20, 100); const filters = { dateFrom: req.query.dateFrom ? new Date(req.query.dateFrom) : undefined, dateTo: req.query.dateTo ? new Date(req.query.dateTo) : undefined, hasIncidents: req.query.hasIncidents === 'true' ? true : req.query.hasIncidents === 'false' ? false : undefined, }; const result = await bitacoraService.findWithFilters(getContext(req), fraccionamientoId, filters, page, limit); res.status(200).json({ success: true, data: result.data, pagination: result.meta, }); } catch (error) { next(error); } }); /** * GET /bitacora/stats * Obtener estadísticas de bitácora */ router.get('/stats', authMiddleware.authenticate, async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const fraccionamientoId = req.query.fraccionamientoId; if (!fraccionamientoId) { res.status(400).json({ error: 'Bad Request', message: 'fraccionamientoId is required' }); return; } const stats = await bitacoraService.getStats(getContext(req), fraccionamientoId); res.status(200).json({ success: true, data: stats }); } catch (error) { next(error); } }); /** * GET /bitacora/latest * Obtener última entrada de bitácora */ router.get('/latest', authMiddleware.authenticate, async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const fraccionamientoId = req.query.fraccionamientoId; if (!fraccionamientoId) { res.status(400).json({ error: 'Bad Request', message: 'fraccionamientoId is required' }); return; } const entry = await bitacoraService.findLatest(getContext(req), fraccionamientoId); if (!entry) { res.status(404).json({ error: 'Not Found', message: 'No log entries found' }); return; } res.status(200).json({ success: true, data: entry }); } catch (error) { next(error); } }); /** * GET /bitacora/:id * Obtener entrada por ID */ router.get('/:id', authMiddleware.authenticate, async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const entry = await bitacoraService.findById(getContext(req), req.params.id); if (!entry) { res.status(404).json({ error: 'Not Found', message: 'Log entry not found' }); return; } res.status(200).json({ success: true, data: entry }); } catch (error) { next(error); } }); /** * POST /bitacora * Crear entrada de bitácora */ router.post('/', authMiddleware.authenticate, authMiddleware.authorize('admin', 'engineer', 'resident'), async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const dto = req.body; if (!dto.fraccionamientoId || !dto.entryDate || !dto.description) { res.status(400).json({ error: 'Bad Request', message: 'fraccionamientoId, entryDate and description are required' }); return; } const entry = await bitacoraService.createEntry(getContext(req), dto); res.status(201).json({ success: true, data: entry }); } catch (error) { next(error); } }); /** * PATCH /bitacora/:id * Actualizar entrada de bitácora */ router.patch('/:id', authMiddleware.authenticate, authMiddleware.authorize('admin', 'engineer', 'resident'), async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const dto = req.body; const entry = await bitacoraService.update(getContext(req), req.params.id, dto); if (!entry) { res.status(404).json({ error: 'Not Found', message: 'Log entry not found' }); return; } res.status(200).json({ success: true, data: entry }); } catch (error) { next(error); } }); /** * DELETE /bitacora/:id * Eliminar entrada de bitácora (soft delete) */ router.delete('/:id', authMiddleware.authenticate, authMiddleware.authorize('admin', 'director'), async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const deleted = await bitacoraService.softDelete(getContext(req), req.params.id); if (!deleted) { res.status(404).json({ error: 'Not Found', message: 'Log entry not found' }); return; } res.status(200).json({ success: true, message: 'Log entry deleted' }); } catch (error) { next(error); } }); return router; } exports.default = createBitacoraObraController; //# sourceMappingURL=bitacora-obra.controller.js.map