360 lines
15 KiB
JavaScript
360 lines
15 KiB
JavaScript
"use strict";
|
|
/**
|
|
* EstimacionController - Controller de estimaciones de obra
|
|
*
|
|
* Endpoints REST para gestión de estimaciones periódicas.
|
|
* Incluye workflow de aprobación: draft -> submitted -> reviewed -> approved -> invoiced -> paid
|
|
*
|
|
* @module Estimates
|
|
*/
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
exports.createEstimacionController = createEstimacionController;
|
|
const express_1 = require("express");
|
|
const estimacion_service_1 = require("../services/estimacion.service");
|
|
const auth_middleware_1 = require("../../auth/middleware/auth.middleware");
|
|
const auth_service_1 = require("../../auth/services/auth.service");
|
|
const estimacion_entity_1 = require("../entities/estimacion.entity");
|
|
const estimacion_concepto_entity_1 = require("../entities/estimacion-concepto.entity");
|
|
const generador_entity_1 = require("../entities/generador.entity");
|
|
const estimacion_workflow_entity_1 = require("../entities/estimacion-workflow.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 estimaciones
|
|
*/
|
|
function createEstimacionController(dataSource) {
|
|
const router = (0, express_1.Router)();
|
|
// Repositorios
|
|
const estimacionRepository = dataSource.getRepository(estimacion_entity_1.Estimacion);
|
|
const conceptoRepository = dataSource.getRepository(estimacion_concepto_entity_1.EstimacionConcepto);
|
|
const generadorRepository = dataSource.getRepository(generador_entity_1.Generador);
|
|
const workflowRepository = dataSource.getRepository(estimacion_workflow_entity_1.EstimacionWorkflow);
|
|
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 estimacionService = new estimacion_service_1.EstimacionService(estimacionRepository, conceptoRepository, generadorRepository, workflowRepository, dataSource);
|
|
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 /estimaciones
|
|
* Listar estimaciones con filtros
|
|
*/
|
|
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 page = parseInt(req.query.page) || 1;
|
|
const limit = Math.min(parseInt(req.query.limit) || 20, 100);
|
|
const filters = {
|
|
contratoId: req.query.contratoId,
|
|
fraccionamientoId: req.query.fraccionamientoId,
|
|
status: req.query.status,
|
|
periodFrom: req.query.periodFrom ? new Date(req.query.periodFrom) : undefined,
|
|
periodTo: req.query.periodTo ? new Date(req.query.periodTo) : undefined,
|
|
};
|
|
const result = await estimacionService.findWithFilters(getContext(req), filters, page, limit);
|
|
res.status(200).json({
|
|
success: true,
|
|
data: result.data,
|
|
pagination: result.meta,
|
|
});
|
|
}
|
|
catch (error) {
|
|
next(error);
|
|
}
|
|
});
|
|
/**
|
|
* GET /estimaciones/summary/:contratoId
|
|
* Obtener resumen de estimaciones por contrato
|
|
*/
|
|
router.get('/summary/:contratoId', 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 summary = await estimacionService.getContractSummary(getContext(req), req.params.contratoId);
|
|
res.status(200).json({ success: true, data: summary });
|
|
}
|
|
catch (error) {
|
|
next(error);
|
|
}
|
|
});
|
|
/**
|
|
* GET /estimaciones/:id
|
|
* Obtener estimación con detalles completos
|
|
*/
|
|
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 estimacion = await estimacionService.findWithDetails(getContext(req), req.params.id);
|
|
if (!estimacion) {
|
|
res.status(404).json({ error: 'Not Found', message: 'Estimate not found' });
|
|
return;
|
|
}
|
|
res.status(200).json({ success: true, data: estimacion });
|
|
}
|
|
catch (error) {
|
|
next(error);
|
|
}
|
|
});
|
|
/**
|
|
* POST /estimaciones
|
|
* Crear estimación
|
|
*/
|
|
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.contratoId || !dto.fraccionamientoId || !dto.periodStart || !dto.periodEnd) {
|
|
res.status(400).json({ error: 'Bad Request', message: 'contratoId, fraccionamientoId, periodStart and periodEnd are required' });
|
|
return;
|
|
}
|
|
const estimacion = await estimacionService.createEstimacion(getContext(req), dto);
|
|
res.status(201).json({ success: true, data: estimacion });
|
|
}
|
|
catch (error) {
|
|
next(error);
|
|
}
|
|
});
|
|
/**
|
|
* POST /estimaciones/:id/conceptos
|
|
* Agregar concepto a estimación
|
|
*/
|
|
router.post('/:id/conceptos', 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.conceptoId || dto.quantityCurrent === undefined || dto.unitPrice === undefined) {
|
|
res.status(400).json({ error: 'Bad Request', message: 'conceptoId, quantityCurrent and unitPrice are required' });
|
|
return;
|
|
}
|
|
const concepto = await estimacionService.addConcepto(getContext(req), req.params.id, dto);
|
|
res.status(201).json({ success: true, data: concepto });
|
|
}
|
|
catch (error) {
|
|
if (error instanceof Error && error.message.includes('non-draft')) {
|
|
res.status(400).json({ error: 'Bad Request', message: error.message });
|
|
return;
|
|
}
|
|
next(error);
|
|
}
|
|
});
|
|
/**
|
|
* POST /estimaciones/conceptos/:conceptoId/generadores
|
|
* Agregar generador a concepto de estimación
|
|
*/
|
|
router.post('/conceptos/:conceptoId/generadores', 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.generatorNumber || dto.quantity === undefined) {
|
|
res.status(400).json({ error: 'Bad Request', message: 'generatorNumber and quantity are required' });
|
|
return;
|
|
}
|
|
const generador = await estimacionService.addGenerador(getContext(req), req.params.conceptoId, dto);
|
|
res.status(201).json({ success: true, data: generador });
|
|
}
|
|
catch (error) {
|
|
if (error instanceof Error && error.message === 'Concepto not found') {
|
|
res.status(404).json({ error: 'Not Found', message: error.message });
|
|
return;
|
|
}
|
|
next(error);
|
|
}
|
|
});
|
|
/**
|
|
* POST /estimaciones/:id/submit
|
|
* Enviar estimación para revisión
|
|
*/
|
|
router.post('/:id/submit', 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 estimacion = await estimacionService.submit(getContext(req), req.params.id);
|
|
if (!estimacion) {
|
|
res.status(400).json({ error: 'Bad Request', message: 'Cannot submit this estimate' });
|
|
return;
|
|
}
|
|
res.status(200).json({ success: true, data: estimacion, message: 'Estimate submitted for review' });
|
|
}
|
|
catch (error) {
|
|
if (error instanceof Error && error.message.includes('Invalid status')) {
|
|
res.status(400).json({ error: 'Bad Request', message: error.message });
|
|
return;
|
|
}
|
|
next(error);
|
|
}
|
|
});
|
|
/**
|
|
* POST /estimaciones/:id/review
|
|
* Revisar estimación
|
|
*/
|
|
router.post('/:id/review', authMiddleware.authenticate, authMiddleware.authorize('admin', 'engineer', '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 estimacion = await estimacionService.review(getContext(req), req.params.id);
|
|
if (!estimacion) {
|
|
res.status(400).json({ error: 'Bad Request', message: 'Cannot review this estimate' });
|
|
return;
|
|
}
|
|
res.status(200).json({ success: true, data: estimacion, message: 'Estimate reviewed' });
|
|
}
|
|
catch (error) {
|
|
if (error instanceof Error && error.message.includes('Invalid status')) {
|
|
res.status(400).json({ error: 'Bad Request', message: error.message });
|
|
return;
|
|
}
|
|
next(error);
|
|
}
|
|
});
|
|
/**
|
|
* POST /estimaciones/:id/approve
|
|
* Aprobar estimación
|
|
*/
|
|
router.post('/:id/approve', 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 estimacion = await estimacionService.approve(getContext(req), req.params.id);
|
|
if (!estimacion) {
|
|
res.status(400).json({ error: 'Bad Request', message: 'Cannot approve this estimate' });
|
|
return;
|
|
}
|
|
res.status(200).json({ success: true, data: estimacion, message: 'Estimate approved' });
|
|
}
|
|
catch (error) {
|
|
if (error instanceof Error && error.message.includes('Invalid status')) {
|
|
res.status(400).json({ error: 'Bad Request', message: error.message });
|
|
return;
|
|
}
|
|
next(error);
|
|
}
|
|
});
|
|
/**
|
|
* POST /estimaciones/:id/reject
|
|
* Rechazar estimación
|
|
*/
|
|
router.post('/:id/reject', authMiddleware.authenticate, authMiddleware.authorize('admin', 'engineer', '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 { reason } = req.body;
|
|
if (!reason) {
|
|
res.status(400).json({ error: 'Bad Request', message: 'reason is required' });
|
|
return;
|
|
}
|
|
const estimacion = await estimacionService.reject(getContext(req), req.params.id, reason);
|
|
if (!estimacion) {
|
|
res.status(400).json({ error: 'Bad Request', message: 'Cannot reject this estimate' });
|
|
return;
|
|
}
|
|
res.status(200).json({ success: true, data: estimacion, message: 'Estimate rejected' });
|
|
}
|
|
catch (error) {
|
|
if (error instanceof Error && error.message.includes('Invalid status')) {
|
|
res.status(400).json({ error: 'Bad Request', message: error.message });
|
|
return;
|
|
}
|
|
next(error);
|
|
}
|
|
});
|
|
/**
|
|
* POST /estimaciones/:id/recalculate
|
|
* Recalcular totales de estimación
|
|
*/
|
|
router.post('/:id/recalculate', authMiddleware.authenticate, authMiddleware.authorize('admin', 'engineer'), async (req, res, next) => {
|
|
try {
|
|
const tenantId = req.tenantId;
|
|
if (!tenantId) {
|
|
res.status(400).json({ error: 'Bad Request', message: 'Tenant ID required' });
|
|
return;
|
|
}
|
|
await estimacionService.recalculateTotals(getContext(req), req.params.id);
|
|
const estimacion = await estimacionService.findWithDetails(getContext(req), req.params.id);
|
|
res.status(200).json({ success: true, data: estimacion, message: 'Totals recalculated' });
|
|
}
|
|
catch (error) {
|
|
next(error);
|
|
}
|
|
});
|
|
/**
|
|
* DELETE /estimaciones/:id
|
|
* Eliminar estimación (solo draft)
|
|
*/
|
|
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 estimacion = await estimacionService.findById(getContext(req), req.params.id);
|
|
if (!estimacion) {
|
|
res.status(404).json({ error: 'Not Found', message: 'Estimate not found' });
|
|
return;
|
|
}
|
|
if (estimacion.status !== 'draft') {
|
|
res.status(400).json({ error: 'Bad Request', message: 'Only draft estimates can be deleted' });
|
|
return;
|
|
}
|
|
const deleted = await estimacionService.softDelete(getContext(req), req.params.id);
|
|
if (!deleted) {
|
|
res.status(404).json({ error: 'Not Found', message: 'Estimate not found' });
|
|
return;
|
|
}
|
|
res.status(200).json({ success: true, message: 'Estimate deleted' });
|
|
}
|
|
catch (error) {
|
|
next(error);
|
|
}
|
|
});
|
|
return router;
|
|
}
|
|
exports.default = createEstimacionController;
|
|
//# sourceMappingURL=estimacion.controller.js.map
|