- Nuevo endpoint GET /reportes/estado-resultados-detallado con cálculo contable:
* Ventas, Devoluciones, Ventas netas, Costo de ventas, Utilidad bruta,
Gastos operativos, Utilidad de la operación
* Fórmula: subtotal_mxn - descuento_mxn (sin impuestos), nómina usa total_mxn
* Excluye anticipos (uso_cfdi=P01 o clave_prod_serv=84111506)
* Filtro por régimen fiscal opcional
* Año anterior calculado automáticamente
- Nuevo endpoint GET /reportes/estado-resultados/drill-down:
* Nivel 1: resumen agrupado por RFC
* Nivel 2: CFDIs individuales filtrados por categoría
* Categorías: ventas, devoluciones, costo-ventas, gastos-operativos
- Nuevo endpoint GET /reportes/estado-resultados/export:
* Genera Excel con formato condicional (verde/rojo, negritas)
- Frontend:
* Tabla vertical con % vertical, año anterior y variación %
* Filas clickeables para drill-down modal de 2 niveles
* Top 10 Clientes/Proveedores mantenidos debajo
* Selector de régimen conectado al reporte
- Fix: NaN en total de drill-down nivel 2 por numeric como string en pg
* Agregado ::float en queries SQL de CFDIs individuales
28 lines
1.4 KiB
TypeScript
28 lines
1.4 KiB
TypeScript
import { Router, type IRouter } from 'express';
|
|
import { authenticate } from '../middlewares/auth.middleware.js';
|
|
import { tenantMiddleware } from '../middlewares/tenant.middleware.js';
|
|
import { checkPlanLimits } from '../middlewares/plan-limits.middleware.js';
|
|
import { requireFeature } from '../middlewares/feature-gate.middleware.js';
|
|
import { normalLimit } from '../middlewares/rate-limit.middleware.js';
|
|
import * as reportesController from '../controllers/reportes.controller.js';
|
|
|
|
const router: IRouter = Router();
|
|
|
|
router.use(authenticate);
|
|
router.use(normalLimit);
|
|
router.use(tenantMiddleware);
|
|
router.use(checkPlanLimits);
|
|
router.use(requireFeature('reportes'));
|
|
|
|
router.get('/estado-resultados', reportesController.getEstadoResultados);
|
|
router.get('/estado-resultados-detallado', reportesController.getEstadoResultadosDetallado);
|
|
router.get('/estado-resultados/drill-down', reportesController.getEstadoResultadosDrillDown);
|
|
router.get('/estado-resultados/export', reportesController.exportEstadoResultados);
|
|
router.get('/flujo-efectivo', reportesController.getFlujoEfectivo);
|
|
router.get('/comparativo', reportesController.getComparativo);
|
|
router.get('/concentrado-rfc', reportesController.getConcentradoRfc);
|
|
router.get('/cuentas-x-pagar', reportesController.getCuentasXPagar);
|
|
router.get('/cuentas-x-cobrar', reportesController.getCuentasXCobrar);
|
|
|
|
export { router as reportesRoutes };
|