"use strict"; /** * ComparativoController - REST API for quotation comparisons * * Endpoints para gestión de cuadros comparativos de cotizaciones. * * @module Purchase * @routes /api/comparativos */ Object.defineProperty(exports, "__esModule", { value: true }); exports.createComparativoController = createComparativoController; const express_1 = require("express"); const comparativo_service_1 = require("../services/comparativo.service"); const comparativo_cotizaciones_entity_1 = require("../entities/comparativo-cotizaciones.entity"); const comparativo_proveedor_entity_1 = require("../entities/comparativo-proveedor.entity"); const comparativo_producto_entity_1 = require("../entities/comparativo-producto.entity"); const auth_middleware_1 = require("../../auth/middleware/auth.middleware"); const auth_service_1 = require("../../auth/services/auth.service"); 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"); function createComparativoController(dataSource) { const router = (0, express_1.Router)(); // Repositories const comparativoRepo = dataSource.getRepository(comparativo_cotizaciones_entity_1.ComparativoCotizaciones); const proveedorRepo = dataSource.getRepository(comparativo_proveedor_entity_1.ComparativoProveedor); const productoRepo = dataSource.getRepository(comparativo_producto_entity_1.ComparativoProducto); 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); // Services const service = new comparativo_service_1.ComparativoService(comparativoRepo, proveedorRepo, productoRepo); const authService = new auth_service_1.AuthService(userRepository, tenantRepository, refreshTokenRepository); const authMiddleware = new auth_middleware_1.AuthMiddleware(authService, dataSource); // Helper for service context const getContext = (req) => { if (!req.tenantId) { throw new Error('Tenant ID is required'); } return { tenantId: req.tenantId, userId: req.user?.sub, }; }; /** * GET /api/comparativos * List comparisons with filters */ 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 filters = {}; if (req.query.requisicionId) { filters.requisicionId = req.query.requisicionId; } if (req.query.status) { filters.status = req.query.status; } if (req.query.dateFrom) { filters.dateFrom = new Date(req.query.dateFrom); } if (req.query.dateTo) { filters.dateTo = new Date(req.query.dateTo); } const page = parseInt(req.query.page) || 1; const limit = Math.min(parseInt(req.query.limit) || 20, 100); const result = await service.findWithFilters(getContext(req), filters, page, limit); res.status(200).json({ success: true, data: result.data, pagination: result.meta }); } catch (error) { next(error); } }); /** * GET /api/comparativos/:id * Get comparison with full details */ 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 comparativo = await service.findWithDetails(getContext(req), req.params.id); if (!comparativo) { res.status(404).json({ error: 'Not Found', message: 'Comparison not found' }); return; } res.status(200).json({ success: true, data: comparativo }); } catch (error) { next(error); } }); /** * POST /api/comparativos * Create new comparison */ router.post('/', authMiddleware.authenticate, authMiddleware.authorize('admin', 'compras'), async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const comparativo = await service.create(getContext(req), { requisicionId: req.body.requisicionId, name: req.body.name, comparisonDate: new Date(req.body.comparisonDate), notes: req.body.notes, }); res.status(201).json({ success: true, data: comparativo }); } catch (error) { next(error); } }); /** * POST /api/comparativos/:id/proveedores * Add supplier to comparison */ router.post('/:id/proveedores', authMiddleware.authenticate, authMiddleware.authorize('admin', 'compras'), async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const proveedor = await service.addProveedor(getContext(req), req.params.id, { supplierId: req.body.supplierId, quotationNumber: req.body.quotationNumber, quotationDate: req.body.quotationDate ? new Date(req.body.quotationDate) : undefined, deliveryDays: req.body.deliveryDays, paymentConditions: req.body.paymentConditions, evaluationNotes: req.body.evaluationNotes, }); res.status(201).json({ success: true, data: proveedor }); } catch (error) { next(error); } }); /** * POST /api/comparativos/proveedores/:proveedorId/productos * Add product to supplier entry */ router.post('/proveedores/:proveedorId/productos', authMiddleware.authenticate, authMiddleware.authorize('admin', 'compras'), async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const producto = await service.addProducto(getContext(req), req.params.proveedorId, { productId: req.body.productId, quantity: parseFloat(req.body.quantity), unitPrice: parseFloat(req.body.unitPrice), notes: req.body.notes, }); res.status(201).json({ success: true, data: producto }); } catch (error) { next(error); } }); /** * PUT /api/comparativos/proveedores/:proveedorId/total * Recalculate supplier total */ router.put('/proveedores/:proveedorId/total', authMiddleware.authenticate, authMiddleware.authorize('admin', 'compras'), async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const proveedor = await service.updateProveedorTotal(getContext(req), req.params.proveedorId); if (!proveedor) { res.status(404).json({ error: 'Not Found', message: 'Supplier entry not found' }); return; } res.status(200).json({ success: true, data: proveedor }); } catch (error) { next(error); } }); /** * POST /api/comparativos/:id/start-evaluation * Start evaluation process */ router.post('/:id/start-evaluation', authMiddleware.authenticate, authMiddleware.authorize('admin', 'compras'), async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const comparativo = await service.startEvaluation(getContext(req), req.params.id); if (!comparativo) { res.status(404).json({ error: 'Not Found', message: 'Comparison not found' }); return; } res.status(200).json({ success: true, data: comparativo }); } catch (error) { next(error); } }); /** * POST /api/comparativos/:id/select-winner * Select winning supplier */ router.post('/:id/select-winner', authMiddleware.authenticate, authMiddleware.authorize('admin', 'compras'), async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const comparativo = await service.selectWinner(getContext(req), req.params.id, req.body.supplierId); if (!comparativo) { res.status(404).json({ error: 'Not Found', message: 'Comparison not found' }); return; } res.status(200).json({ success: true, data: comparativo }); } catch (error) { next(error); } }); /** * POST /api/comparativos/:id/cancel * Cancel comparison */ router.post('/:id/cancel', authMiddleware.authenticate, authMiddleware.authorize('admin', 'compras'), async (req, res, next) => { try { const tenantId = req.tenantId; if (!tenantId) { res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' }); return; } const comparativo = await service.cancel(getContext(req), req.params.id); if (!comparativo) { res.status(404).json({ error: 'Not Found', message: 'Comparison not found' }); return; } res.status(200).json({ success: true, data: comparativo }); } catch (error) { next(error); } }); /** * DELETE /api/comparativos/:id * Soft delete comparison */ router.delete('/:id', authMiddleware.authenticate, authMiddleware.authorize('admin'), 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 service.softDelete(getContext(req), req.params.id); if (!deleted) { res.status(404).json({ error: 'Not Found', message: 'Comparison not found' }); return; } res.status(204).send(); } catch (error) { next(error); } }); return router; } //# sourceMappingURL=comparativo.controller.js.map