erp-construccion-backend/dist/modules/purchase/controllers/comparativo.controller.js

283 lines
11 KiB
JavaScript

"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