feat: restaurar panorama facturador con nueva arquitectura basada en Metabase
- Crear endpoint /api/metabase/panorama.post.ts que ejecuta las 9 queries en paralelo - Restaurar y adaptar panorama.vue para usar el nuevo endpoint - Crear componentes auxiliares: SecosVendidos, TotalesIngresoCompra, TotalesMonetarios, TotalesVerde, MetricBox, RechazosRechazoCard - Adaptar RechazosSubproductos para recibir data directamente de Metabase - Toda la transformación de datos ocurre en las queries SQL de Metabase - Sin uso de stores ni composables de métricas - Agregar documentación de queries en archivos MD
This commit is contained in:
555
METABASE_QUERIES_COMPARATIVA_COSECHAS.md
Normal file
555
METABASE_QUERIES_COMPARATIVA_COSECHAS.md
Normal file
@@ -0,0 +1,555 @@
|
||||
# 📊 QUERIES DE METABASE PARA COMPARATIVA DE COSECHAS
|
||||
|
||||
## 🎯 FILOSOFÍA
|
||||
|
||||
**Metabase calcula TODO. Vue solo renderiza.**
|
||||
|
||||
Esta página trabaja con **`vista_resumen_ingresos`** (datos agregados por día), NO con `vista_detalle_ingresos`.
|
||||
|
||||
---
|
||||
|
||||
## 📋 PARÁMETROS GLOBALES
|
||||
|
||||
Todas las queries aceptan estos parámetros:
|
||||
|
||||
| Parámetro | Tipo | Default | Descripción |
|
||||
|-----------|------|---------|-------------|
|
||||
| `cosechas_ids` | Array[String] | `[]` | Array de IDs de cosechas: 'cosecha-20-21', 'cosecha-21-22', etc. |
|
||||
| `incluir_anulados` | Boolean | `false` | Si `false`, excluye registros anulados |
|
||||
|
||||
**IMPORTANTE:** En lugar de fechas libres, se usan IDs de cosechas que se mapean a rangos predefinidos en el backend.
|
||||
|
||||
---
|
||||
|
||||
## 📅 DEFINICIÓN DE COSECHAS (HARDCODED)
|
||||
|
||||
Estas definiciones están hardcoded en el código pero necesitan estar también en las queries de Metabase:
|
||||
|
||||
```sql
|
||||
-- Mapping de cosechas a rangos de fechas
|
||||
CASE
|
||||
WHEN 'cosecha-20-21' = ANY({{cosechas_ids}}) AND fecha >= '2020-09-08' AND fecha <= '2021-09-07' THEN 'cosecha-20-21'
|
||||
WHEN 'cosecha-21-22' = ANY({{cosechas_ids}}) AND fecha >= '2021-09-08' AND fecha <= '2022-09-07' THEN 'cosecha-21-22'
|
||||
WHEN 'cosecha-22-23' = ANY({{cosechas_ids}}) AND fecha >= '2022-09-08' AND fecha <= '2023-09-07' THEN 'cosecha-22-23'
|
||||
WHEN 'cosecha-23-24' = ANY({{cosechas_ids}}) AND fecha >= '2023-09-08' AND fecha <= '2024-09-07' THEN 'cosecha-23-24'
|
||||
WHEN 'cosecha-24-25' = ANY({{cosechas_ids}}) AND fecha >= '2024-09-08' AND fecha <= '2025-09-07' THEN 'cosecha-24-25'
|
||||
WHEN 'cosecha-25-26' = ANY({{cosechas_ids}}) AND fecha >= '2025-09-08' THEN 'cosecha-25-26'
|
||||
ELSE NULL
|
||||
END as cosecha_id
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 🎨 QUERIES POR COMPONENTE
|
||||
|
||||
### Query 1: `comparativa_datos_diarios_completos`
|
||||
**Componente:** `CosechasHeatmap.vue`, `CosechasEvolucion.vue`
|
||||
**Devuelve:** Múltiples filas (una por día de cada cosecha seleccionada)
|
||||
|
||||
```sql
|
||||
WITH cosechas_mapping AS (
|
||||
SELECT
|
||||
fecha,
|
||||
-- Identificar a qué cosecha pertenece cada fecha
|
||||
CASE
|
||||
WHEN fecha >= '2020-09-08' AND fecha <= '2021-09-07' THEN 'cosecha-20-21'
|
||||
WHEN fecha >= '2021-09-08' AND fecha <= '2022-09-07' THEN 'cosecha-21-22'
|
||||
WHEN fecha >= '2022-09-08' AND fecha <= '2023-09-07' THEN 'cosecha-22-23'
|
||||
WHEN fecha >= '2023-09-08' AND fecha <= '2024-09-07' THEN 'cosecha-23-24'
|
||||
WHEN fecha >= '2024-09-08' AND fecha <= '2025-09-07' THEN 'cosecha-24-25'
|
||||
WHEN fecha >= '2025-09-08' THEN 'cosecha-25-26'
|
||||
ELSE NULL
|
||||
END as cosecha_id,
|
||||
|
||||
-- Calcular día relativo dentro de cada cosecha (empieza en 1)
|
||||
CASE
|
||||
WHEN fecha >= '2020-09-08' AND fecha <= '2021-09-07'
|
||||
THEN (fecha - DATE '2020-09-08') + 1
|
||||
WHEN fecha >= '2021-09-08' AND fecha <= '2022-09-07'
|
||||
THEN (fecha - DATE '2021-09-08') + 1
|
||||
WHEN fecha >= '2022-09-08' AND fecha <= '2023-09-07'
|
||||
THEN (fecha - DATE '2022-09-08') + 1
|
||||
WHEN fecha >= '2023-09-08' AND fecha <= '2024-09-07'
|
||||
THEN (fecha - DATE '2023-09-08') + 1
|
||||
WHEN fecha >= '2024-09-08' AND fecha <= '2025-09-07'
|
||||
THEN (fecha - DATE '2024-09-08') + 1
|
||||
WHEN fecha >= '2025-09-08'
|
||||
THEN (fecha - DATE '2025-09-08') + 1
|
||||
ELSE NULL
|
||||
END as dia_relativo,
|
||||
|
||||
-- Métricas del día
|
||||
COALESCE(total_peso_seco, 0) as total_peso_seco,
|
||||
COALESCE(peso_neto_uva, 0) as peso_neto_uva,
|
||||
COALESCE(peso_neto_verde, 0) as peso_neto_verde,
|
||||
COALESCE(sacos_total_dia, 0) as sacos_total_dia,
|
||||
COALESCE(total_lempiras_uva, 0) as total_lempiras_uva,
|
||||
COALESCE(total_lempiras_verde, 0) as total_lempiras_verde,
|
||||
COALESCE(total_lempiras_mojado, 0) as total_lempiras_mojado,
|
||||
COALESCE(total_lempiras_oreado, 0) as total_lempiras_oreado,
|
||||
COALESCE(total_lempiras_mojado, 0) + COALESCE(total_lempiras_oreado, 0) as total_lempiras_mojado_oreado
|
||||
|
||||
FROM vista_resumen_ingresos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
-- Filtrar solo las fechas que pertenecen a cosechas seleccionadas
|
||||
AND (
|
||||
(fecha >= '2020-09-08' AND fecha <= '2021-09-07' AND 'cosecha-20-21' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2021-09-08' AND fecha <= '2022-09-07' AND 'cosecha-21-22' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2022-09-08' AND fecha <= '2023-09-07' AND 'cosecha-22-23' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2023-09-08' AND fecha <= '2024-09-07' AND 'cosecha-23-24' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2024-09-08' AND fecha <= '2025-09-07' AND 'cosecha-24-25' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2025-09-08' AND 'cosecha-25-26' = ANY({{cosechas_ids}}))
|
||||
)
|
||||
)
|
||||
SELECT
|
||||
fecha,
|
||||
cosecha_id,
|
||||
dia_relativo,
|
||||
total_peso_seco,
|
||||
peso_neto_uva,
|
||||
peso_neto_verde,
|
||||
sacos_total_dia,
|
||||
total_lempiras_uva,
|
||||
total_lempiras_verde,
|
||||
total_lempiras_mojado,
|
||||
total_lempiras_oreado,
|
||||
total_lempiras_mojado_oreado
|
||||
FROM cosechas_mapping
|
||||
WHERE cosecha_id IS NOT NULL
|
||||
ORDER BY cosecha_id, fecha;
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
[
|
||||
{
|
||||
"fecha": "2025-01-15",
|
||||
"cosecha_id": "cosecha-24-25",
|
||||
"dia_relativo": 130,
|
||||
"total_peso_seco": 45.50,
|
||||
"peso_neto_uva": 2250.00,
|
||||
"peso_neto_verde": 500.00,
|
||||
"sacos_total_dia": 50,
|
||||
"total_lempiras_uva": 11475.00,
|
||||
"total_lempiras_verde": 5000.00,
|
||||
"total_lempiras_mojado": 8000.00,
|
||||
"total_lempiras_oreado": 6000.00,
|
||||
"total_lempiras_mojado_oreado": 14000.00
|
||||
}
|
||||
]
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 2: `comparativa_totales_por_cosecha`
|
||||
**Componente:** `CosechasTotales.vue`
|
||||
**Devuelve:** Múltiples filas (una por cosecha seleccionada con totales agregados)
|
||||
|
||||
```sql
|
||||
WITH cosechas_mapping AS (
|
||||
SELECT
|
||||
CASE
|
||||
WHEN fecha >= '2020-09-08' AND fecha <= '2021-09-07' THEN 'cosecha-20-21'
|
||||
WHEN fecha >= '2021-09-08' AND fecha <= '2022-09-07' THEN 'cosecha-21-22'
|
||||
WHEN fecha >= '2022-09-08' AND fecha <= '2023-09-07' THEN 'cosecha-22-23'
|
||||
WHEN fecha >= '2023-09-08' AND fecha <= '2024-09-07' THEN 'cosecha-23-24'
|
||||
WHEN fecha >= '2024-09-08' AND fecha <= '2025-09-07' THEN 'cosecha-24-25'
|
||||
WHEN fecha >= '2025-09-08' THEN 'cosecha-25-26'
|
||||
ELSE NULL
|
||||
END as cosecha_id,
|
||||
|
||||
total_peso_seco,
|
||||
peso_neto_uva,
|
||||
peso_neto_verde,
|
||||
sacos_total_dia,
|
||||
total_lempiras_uva,
|
||||
total_lempiras_verde,
|
||||
total_lempiras_mojado,
|
||||
total_lempiras_oreado
|
||||
|
||||
FROM vista_resumen_ingresos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND (
|
||||
(fecha >= '2020-09-08' AND fecha <= '2021-09-07' AND 'cosecha-20-21' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2021-09-08' AND fecha <= '2022-09-07' AND 'cosecha-21-22' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2022-09-08' AND fecha <= '2023-09-07' AND 'cosecha-22-23' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2023-09-08' AND fecha <= '2024-09-07' AND 'cosecha-23-24' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2024-09-08' AND fecha <= '2025-09-07' AND 'cosecha-24-25' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2025-09-08' AND 'cosecha-25-26' = ANY({{cosechas_ids}}))
|
||||
)
|
||||
)
|
||||
SELECT
|
||||
cosecha_id,
|
||||
|
||||
-- Totales agregados
|
||||
COUNT(*) as num_dias,
|
||||
COALESCE(SUM(total_peso_seco), 0) as peso_total,
|
||||
COALESCE(SUM(peso_neto_uva), 0) as peso_uva_total,
|
||||
COALESCE(SUM(peso_neto_verde), 0) as peso_verde_total,
|
||||
COALESCE(SUM(sacos_total_dia), 0) as sacos_total,
|
||||
|
||||
-- Inversiones totales
|
||||
COALESCE(SUM(total_lempiras_uva), 0) as inversion_uva_total,
|
||||
COALESCE(SUM(total_lempiras_verde), 0) as inversion_verde_total,
|
||||
COALESCE(SUM(total_lempiras_mojado), 0) as inversion_mojado_total,
|
||||
COALESCE(SUM(total_lempiras_oreado), 0) as inversion_oreado_total,
|
||||
COALESCE(SUM(total_lempiras_uva), 0) +
|
||||
COALESCE(SUM(total_lempiras_verde), 0) +
|
||||
COALESCE(SUM(total_lempiras_mojado), 0) +
|
||||
COALESCE(SUM(total_lempiras_oreado), 0) as inversion_total,
|
||||
|
||||
-- Promedios diarios
|
||||
AVG(total_peso_seco) as peso_promedio_diario,
|
||||
AVG(sacos_total_dia) as sacos_promedio_diario
|
||||
|
||||
FROM cosechas_mapping
|
||||
WHERE cosecha_id IS NOT NULL
|
||||
GROUP BY cosecha_id
|
||||
ORDER BY cosecha_id;
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
[
|
||||
{
|
||||
"cosecha_id": "cosecha-23-24",
|
||||
"num_dias": 365,
|
||||
"peso_total": 16425.50,
|
||||
"peso_uva_total": 10000.00,
|
||||
"peso_verde_total": 3000.00,
|
||||
"sacos_total": 18250,
|
||||
"inversion_uva_total": 4500000.00,
|
||||
"inversion_verde_total": 300000.00,
|
||||
"inversion_mojado_total": 800000.00,
|
||||
"inversion_oreado_total": 600000.00,
|
||||
"inversion_total": 6200000.00,
|
||||
"peso_promedio_diario": 45.00,
|
||||
"sacos_promedio_diario": 50.00
|
||||
},
|
||||
{
|
||||
"cosecha_id": "cosecha-24-25",
|
||||
"num_dias": 130,
|
||||
"peso_total": 5850.00,
|
||||
"peso_uva_total": 3500.00,
|
||||
"peso_verde_total": 1000.00,
|
||||
"sacos_total": 6500,
|
||||
"inversion_uva_total": 1575000.00,
|
||||
"inversion_verde_total": 100000.00,
|
||||
"inversion_mojado_total": 280000.00,
|
||||
"inversion_oreado_total": 210000.00,
|
||||
"inversion_total": 2165000.00,
|
||||
"peso_promedio_diario": 45.00,
|
||||
"sacos_promedio_diario": 50.00
|
||||
}
|
||||
]
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 3: `comparativa_datos_acumulados_por_dia`
|
||||
**Componente:** `CosechasEvolucion.vue` (modo acumulado)
|
||||
**Devuelve:** Múltiples filas (una por día con valores acumulados)
|
||||
|
||||
```sql
|
||||
WITH cosechas_mapping AS (
|
||||
SELECT
|
||||
fecha,
|
||||
CASE
|
||||
WHEN fecha >= '2020-09-08' AND fecha <= '2021-09-07' THEN 'cosecha-20-21'
|
||||
WHEN fecha >= '2021-09-08' AND fecha <= '2022-09-07' THEN 'cosecha-21-22'
|
||||
WHEN fecha >= '2022-09-08' AND fecha <= '2023-09-07' THEN 'cosecha-22-23'
|
||||
WHEN fecha >= '2023-09-08' AND fecha <= '2024-09-07' THEN 'cosecha-23-24'
|
||||
WHEN fecha >= '2024-09-08' AND fecha <= '2025-09-07' THEN 'cosecha-24-25'
|
||||
WHEN fecha >= '2025-09-08' THEN 'cosecha-25-26'
|
||||
ELSE NULL
|
||||
END as cosecha_id,
|
||||
|
||||
CASE
|
||||
WHEN fecha >= '2020-09-08' AND fecha <= '2021-09-07'
|
||||
THEN (fecha - DATE '2020-09-08') + 1
|
||||
WHEN fecha >= '2021-09-08' AND fecha <= '2022-09-07'
|
||||
THEN (fecha - DATE '2021-09-08') + 1
|
||||
WHEN fecha >= '2022-09-08' AND fecha <= '2023-09-07'
|
||||
THEN (fecha - DATE '2022-09-08') + 1
|
||||
WHEN fecha >= '2023-09-08' AND fecha <= '2024-09-07'
|
||||
THEN (fecha - DATE '2023-09-08') + 1
|
||||
WHEN fecha >= '2024-09-08' AND fecha <= '2025-09-07'
|
||||
THEN (fecha - DATE '2024-09-08') + 1
|
||||
WHEN fecha >= '2025-09-08'
|
||||
THEN (fecha - DATE '2025-09-08') + 1
|
||||
ELSE NULL
|
||||
END as dia_relativo,
|
||||
|
||||
total_peso_seco
|
||||
|
||||
FROM vista_resumen_ingresos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND (
|
||||
(fecha >= '2020-09-08' AND fecha <= '2021-09-07' AND 'cosecha-20-21' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2021-09-08' AND fecha <= '2022-09-07' AND 'cosecha-21-22' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2022-09-08' AND fecha <= '2023-09-07' AND 'cosecha-22-23' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2023-09-08' AND fecha <= '2024-09-07' AND 'cosecha-23-24' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2024-09-08' AND fecha <= '2025-09-07' AND 'cosecha-24-25' = ANY({{cosechas_ids}}))
|
||||
OR (fecha >= '2025-09-08' AND 'cosecha-25-26' = ANY({{cosechas_ids}}))
|
||||
)
|
||||
)
|
||||
SELECT
|
||||
fecha,
|
||||
cosecha_id,
|
||||
dia_relativo,
|
||||
total_peso_seco as peso_dia,
|
||||
|
||||
-- Acumulado hasta este día (dentro de cada cosecha)
|
||||
SUM(total_peso_seco) OVER (
|
||||
PARTITION BY cosecha_id
|
||||
ORDER BY fecha
|
||||
ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW
|
||||
) as peso_acumulado
|
||||
|
||||
FROM cosechas_mapping
|
||||
WHERE cosecha_id IS NOT NULL
|
||||
ORDER BY cosecha_id, fecha;
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
[
|
||||
{
|
||||
"fecha": "2024-09-08",
|
||||
"cosecha_id": "cosecha-24-25",
|
||||
"dia_relativo": 1,
|
||||
"peso_dia": 45.50,
|
||||
"peso_acumulado": 45.50
|
||||
},
|
||||
{
|
||||
"fecha": "2024-09-09",
|
||||
"cosecha_id": "cosecha-24-25",
|
||||
"dia_relativo": 2,
|
||||
"peso_dia": 50.00,
|
||||
"peso_acumulado": 95.50
|
||||
}
|
||||
]
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 4: `comparativa_metadata_cosechas`
|
||||
**Propósito:** Obtener información de disponibilidad de cada cosecha (cuántos días tiene datos)
|
||||
**Devuelve:** Múltiples filas (una por cosecha)
|
||||
|
||||
```sql
|
||||
WITH cosechas_definidas AS (
|
||||
SELECT unnest(ARRAY[
|
||||
'cosecha-20-21',
|
||||
'cosecha-21-22',
|
||||
'cosecha-22-23',
|
||||
'cosecha-23-24',
|
||||
'cosecha-24-25',
|
||||
'cosecha-25-26'
|
||||
]) as cosecha_id
|
||||
),
|
||||
cosechas_mapping AS (
|
||||
SELECT
|
||||
CASE
|
||||
WHEN fecha >= '2020-09-08' AND fecha <= '2021-09-07' THEN 'cosecha-20-21'
|
||||
WHEN fecha >= '2021-09-08' AND fecha <= '2022-09-07' THEN 'cosecha-21-22'
|
||||
WHEN fecha >= '2022-09-08' AND fecha <= '2023-09-07' THEN 'cosecha-22-23'
|
||||
WHEN fecha >= '2023-09-08' AND fecha <= '2024-09-07' THEN 'cosecha-23-24'
|
||||
WHEN fecha >= '2024-09-08' AND fecha <= '2025-09-07' THEN 'cosecha-24-25'
|
||||
WHEN fecha >= '2025-09-08' THEN 'cosecha-25-26'
|
||||
ELSE NULL
|
||||
END as cosecha_id,
|
||||
fecha
|
||||
FROM vista_resumen_ingresos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
)
|
||||
SELECT
|
||||
d.cosecha_id,
|
||||
COALESCE(COUNT(m.fecha), 0) as num_registros,
|
||||
MIN(m.fecha) as primera_fecha,
|
||||
MAX(m.fecha) as ultima_fecha,
|
||||
|
||||
-- Verificar si tiene datos hasta hoy
|
||||
CASE
|
||||
WHEN MAX(m.fecha) >= CURRENT_DATE THEN true
|
||||
ELSE false
|
||||
END as tiene_datos_hasta_hoy
|
||||
|
||||
FROM cosechas_definidas d
|
||||
LEFT JOIN cosechas_mapping m ON d.cosecha_id = m.cosecha_id
|
||||
GROUP BY d.cosecha_id
|
||||
ORDER BY d.cosecha_id;
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
[
|
||||
{ "cosecha_id": "cosecha-20-21", "num_registros": 200, "primera_fecha": "2020-09-08", "ultima_fecha": "2021-09-07", "tiene_datos_hasta_hoy": false },
|
||||
{ "cosecha_id": "cosecha-21-22", "num_registros": 0, "primera_fecha": null, "ultima_fecha": null, "tiene_datos_hasta_hoy": false },
|
||||
{ "cosecha_id": "cosecha-22-23", "num_registros": 300, "primera_fecha": "2022-09-08", "ultima_fecha": "2023-09-07", "tiene_datos_hasta_hoy": false },
|
||||
{ "cosecha_id": "cosecha-23-24", "num_registros": 365, "primera_fecha": "2023-09-08", "ultima_fecha": "2024-09-07", "tiene_datos_hasta_hoy": false },
|
||||
{ "cosecha_id": "cosecha-24-25", "num_registros": 130, "primera_fecha": "2024-09-08", "ultima_fecha": "2025-01-15", "tiene_datos_hasta_hoy": true },
|
||||
{ "cosecha_id": "cosecha-25-26", "num_registros": 0, "primera_fecha": null, "ultima_fecha": null, "tiene_datos_hasta_hoy": false }
|
||||
]
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 📊 RESUMEN DE QUERIES
|
||||
|
||||
| # | Nombre | Componente | Tipo de respuesta |
|
||||
|---|--------|------------|-------------------|
|
||||
| 1 | `comparativa_datos_diarios_completos` | Heatmap, Evolución | Múltiples filas (días) |
|
||||
| 2 | `comparativa_totales_por_cosecha` | Totales (barras) | Múltiples filas (cosechas) |
|
||||
| 3 | `comparativa_datos_acumulados_por_dia` | Evolución (acumulado) | Múltiples filas (días) |
|
||||
| 4 | `comparativa_metadata_cosechas` | Selector de cosechas | Múltiples filas (cosechas) |
|
||||
|
||||
**Total: 4 Queries**
|
||||
|
||||
---
|
||||
|
||||
## 🔧 CONFIGURACIÓN DE PARÁMETROS EN METABASE
|
||||
|
||||
### `cosechas_ids`
|
||||
- **Tipo:** Text (se parsea como array)
|
||||
- **Widget Type:** Text
|
||||
- **Default:** `[]`
|
||||
- **Required:** No
|
||||
- **SQL Type:** `TEXT[]`
|
||||
- **Valores posibles:** `['cosecha-20-21', 'cosecha-21-22', 'cosecha-22-23', 'cosecha-23-24', 'cosecha-24-25', 'cosecha-25-26']`
|
||||
|
||||
### `incluir_anulados`
|
||||
- **Tipo:** Boolean
|
||||
- **Widget Type:** Boolean
|
||||
- **Default:** `false`
|
||||
- **Required:** No
|
||||
|
||||
---
|
||||
|
||||
## 🚀 INTEGRACIÓN EN VUE
|
||||
|
||||
```typescript
|
||||
// comparativa-cosechas.vue
|
||||
|
||||
const cosechasSeleccionadas = ref<string[]>(['cosecha-24-25', 'cosecha-25-26'])
|
||||
const incluirAnulados = ref(false)
|
||||
|
||||
const params = computed(() => ({
|
||||
cosechas_ids: cosechasSeleccionadas.value,
|
||||
incluir_anulados: incluirAnulados.value
|
||||
}))
|
||||
|
||||
async function loadAllData() {
|
||||
const [
|
||||
datosDiarios,
|
||||
totalesCosechas,
|
||||
datosAcumulados,
|
||||
metadataCosechas
|
||||
] = await Promise.all([
|
||||
$fetch('/api/metabase/question/201', { query: params.value }),
|
||||
$fetch('/api/metabase/question/202', { query: params.value }),
|
||||
$fetch('/api/metabase/question/203', { query: params.value }),
|
||||
$fetch('/api/metabase/question/204', { query: params.value })
|
||||
])
|
||||
|
||||
// Asignar directamente
|
||||
datos.value = {
|
||||
diarios: datosDiarios.data.rows,
|
||||
totales: totalesCosechas.data.rows,
|
||||
acumulados: datosAcumulados.data.rows,
|
||||
metadata: metadataCosechas.data.rows
|
||||
}
|
||||
}
|
||||
|
||||
// Transformar para componentes
|
||||
const resumenIngresos = computed(() => {
|
||||
// Convertir datos.value.diarios al formato que esperan los componentes
|
||||
return datos.value.diarios.map(row => ({
|
||||
fecha: row.fecha,
|
||||
cosecha_id: row.cosecha_id,
|
||||
dia_relativo: row.dia_relativo,
|
||||
total_peso_seco: row.total_peso_seco,
|
||||
peso_neto_uva: row.peso_neto_uva,
|
||||
peso_neto_verde: row.peso_neto_verde,
|
||||
sacos_total_dia: row.sacos_total_dia,
|
||||
total_lempiras_uva: row.total_lempiras_uva,
|
||||
total_lempiras_verde: row.total_lempiras_verde,
|
||||
total_lempiras_mojado: row.total_lempiras_mojado,
|
||||
total_lempiras_oreado: row.total_lempiras_oreado,
|
||||
total_lempiras_mojado_oreado: row.total_lempiras_mojado_oreado
|
||||
}))
|
||||
})
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## ⚠️ NOTAS IMPORTANTES
|
||||
|
||||
### 1. Fuente de Datos Diferente
|
||||
|
||||
Esta página usa **`vista_resumen_ingresos`**, NO `vista_detalle_ingresos`:
|
||||
- **Granularidad:** Por DÍA (un registro = un día completo con totales)
|
||||
- **Estructura:** Ya viene agregada con totales diarios
|
||||
- **Performance:** Mucho más rápida porque trabaja con menos registros
|
||||
|
||||
### 2. Cosechas Hardcoded
|
||||
|
||||
Los rangos de cosechas están hardcoded:
|
||||
- **Cosecha X-Y:** Del 8 de septiembre del año X al 7 de septiembre del año Y
|
||||
- **Última cosecha:** Del 8 de septiembre hasta "hoy"
|
||||
|
||||
Si se agregan nuevas cosechas, hay que actualizar:
|
||||
- Las queries de Metabase
|
||||
- La definición en el código Vue (`cosechasDefiniciones`)
|
||||
|
||||
### 3. Día Relativo
|
||||
|
||||
Cada cosecha tiene su propio "día relativo" que empieza en 1:
|
||||
- **Día 1:** 8 de septiembre (inicio de cosecha)
|
||||
- **Día 365:** 7 de septiembre del año siguiente (fin de cosecha)
|
||||
|
||||
Esto permite comparar "día a día" entre cosechas diferentes.
|
||||
|
||||
### 4. Métricas Computadas
|
||||
|
||||
La métrica `total_lempiras_mojado_oreado` es computada:
|
||||
```sql
|
||||
COALESCE(total_lempiras_mojado, 0) + COALESCE(total_lempiras_oreado, 0)
|
||||
```
|
||||
|
||||
### 5. Performance
|
||||
|
||||
- Query 1 puede devolver hasta ~2000 filas (365 días × 6 cosechas max)
|
||||
- Query 3 usa window functions, requiere índice en `fecha`
|
||||
- Recomendado: Crear índices en `vista_resumen_ingresos(fecha, estado)`
|
||||
|
||||
### 6. Heatmap y Evolución
|
||||
|
||||
Ambos componentes usan la **misma Query 1**, pero procesan los datos diferente:
|
||||
- **Heatmap:** Muestra matriz de días × cosechas con colores
|
||||
- **Evolución:** Muestra gráficos de líneas temporales
|
||||
|
||||
Por eso Query 1 debe devolver TODOS los campos necesarios.
|
||||
|
||||
---
|
||||
|
||||
## 📝 PRÓXIMOS PASOS
|
||||
|
||||
1. ✅ Crear estas 4 queries en Metabase
|
||||
2. ✅ Configurar parámetros (especialmente `cosechas_ids` como array)
|
||||
3. ✅ Probar con diferentes combinaciones de cosechas
|
||||
4. ✅ Obtener los Question IDs
|
||||
5. ✅ Adaptar `comparativa-cosechas.vue` para usar las nuevas queries
|
||||
6. ✅ Crear función de transformación de datos para los componentes
|
||||
7. ✅ Probar performance con datasets completos (365 días × N cosechas)
|
||||
8. ✅ Verificar que el heatmap y evolución funcionen correctamente
|
||||
|
||||
---
|
||||
|
||||
**Documento creado:** 2025-10-14
|
||||
**Autor:** Claude Code
|
||||
**Proyecto:** Analítica Núcleo - Comparativa de Cosechas
|
||||
686
METABASE_QUERIES_INFORME_INGRESOS.md
Normal file
686
METABASE_QUERIES_INFORME_INGRESOS.md
Normal file
@@ -0,0 +1,686 @@
|
||||
# 📊 QUERIES DE METABASE PARA INFORME INGRESOS
|
||||
|
||||
## 🎯 FILOSOFÍA
|
||||
|
||||
**Metabase calcula TODO. Vue solo renderiza.**
|
||||
|
||||
Cada query devuelve exactamente los valores que un componente necesita mostrar, ya calculados y listos para usar.
|
||||
|
||||
---
|
||||
|
||||
## 📋 PARÁMETROS GLOBALES
|
||||
|
||||
Todas las queries aceptan estos parámetros (con soporte para múltiples valores):
|
||||
|
||||
| Parámetro | Tipo | Default | Descripción |
|
||||
|-----------|------|---------|-------------|
|
||||
| `fecha_desde` | Date | `null` | Fecha inicio (filtra por `created_at`) |
|
||||
| `fecha_hasta` | Date | `null` | Fecha fin (filtra por `created_at`) |
|
||||
| `incluir_anulados` | Boolean | `false` | Si `false`, excluye `estado='anulado'` o `fecha_anulado IS NOT NULL` |
|
||||
| `cliente_ids` | Array[Number] | `[]` | Array de IDs de clientes. Si vacío, incluye todos |
|
||||
| `tipos` | Array[String] | `[]` | Array de tipos: 'uva', 'mojado', 'oreado', 'verde'. Si vacío, todos |
|
||||
| `estados` | Array[String] | `[]` | Array de estados: 'pagado', 'pendiente'. Si vacío, todos |
|
||||
| `ubicaciones` | Array[String] | `[]` | Array de ubicaciones de clientes. Si vacío, todas |
|
||||
| `calidades` | Array[String] | `[]` | Array de calidades de café. Si vacío, todas |
|
||||
|
||||
**Condición SQL estándar:**
|
||||
```sql
|
||||
WHERE
|
||||
-- Filtro de anulados
|
||||
({{incluir_anulados}} OR (i.estado != 'anulado' AND i.fecha_anulado IS NULL))
|
||||
|
||||
-- Filtro de fechas
|
||||
AND ({{fecha_desde}} IS NULL OR i.created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR i.created_at <= {{fecha_hasta}})
|
||||
|
||||
-- Filtro de clientes (si se proporciona)
|
||||
AND (CARDINALITY({{cliente_ids}}) = 0 OR i.cliente_id = ANY({{cliente_ids}}))
|
||||
|
||||
-- Filtro de tipos (si se proporciona)
|
||||
AND (CARDINALITY({{tipos}}) = 0 OR i.tipo = ANY({{tipos}}))
|
||||
|
||||
-- Filtro de estados (si se proporciona)
|
||||
AND (CARDINALITY({{estados}}) = 0 OR i.estado = ANY({{estados}}))
|
||||
|
||||
-- Filtro de ubicaciones vía join con clientes (si se proporciona)
|
||||
AND (CARDINALITY({{ubicaciones}}) = 0 OR c.ubicacion = ANY({{ubicaciones}}))
|
||||
|
||||
-- Filtro de calidades (si se proporciona)
|
||||
AND (CARDINALITY({{calidades}}) = 0 OR i.calidad = ANY({{calidades}}))
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 🔁 QUERIES REUTILIZADAS DE PANORAMA
|
||||
|
||||
Las siguientes queries son **idénticas** a las de panorama, solo cambia que aceptan más parámetros de filtrado:
|
||||
|
||||
### Query 1: `informe_totales_ingreso_compra`
|
||||
**Reutiliza:** `panorama_totales_ingreso_compra`
|
||||
**Componente:** `TotalesIngresoCompra.vue`
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
-- === TOTALES GENERALES (Pagado + Pendiente) ===
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'uva' THEN i.peso_neto ELSE 0 END), 0) as total_lb_uva_ingresada,
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'uva' THEN i.peso_seco ELSE 0 END), 0) as total_qq_seco_uva_ingresado,
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'mojado' THEN i.peso_seco ELSE 0 END), 0) as total_qq_seco_mojado_ingresado,
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'oreado' THEN i.peso_seco ELSE 0 END), 0) as total_qq_seco_oreado_ingresado,
|
||||
COALESCE(SUM(i.peso_seco), 0) as total_qq_seco_ingresado,
|
||||
|
||||
-- === SOLO PAGADOS ===
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'uva' AND i.estado = 'pagado' THEN i.peso_neto ELSE 0 END), 0) as total_lb_uva_pagada,
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'uva' AND i.estado = 'pagado' THEN i.peso_seco ELSE 0 END), 0) as total_qq_seco_uva_pagado,
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'mojado' AND i.estado = 'pagado' THEN i.peso_seco ELSE 0 END), 0) as total_qq_seco_mojado_pagado,
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'oreado' AND i.estado = 'pagado' THEN i.peso_seco ELSE 0 END), 0) as total_qq_seco_oreado_pagado,
|
||||
COALESCE(SUM(CASE WHEN i.estado = 'pagado' THEN i.peso_seco ELSE 0 END), 0) as total_qq_seco_comprado,
|
||||
|
||||
-- === INVENTARIO EN DEPÓSITO (Pendiente) ===
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'uva' AND i.estado = 'pendiente' THEN i.peso_neto ELSE 0 END), 0) as total_lb_uva_deposito,
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'mojado' AND i.estado = 'pendiente' THEN i.peso_seco ELSE 0 END), 0) as total_qq_mojado_deposito,
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'oreado' AND i.estado = 'pendiente' THEN i.peso_seco ELSE 0 END), 0) as total_qq_oreado_deposito,
|
||||
COALESCE(SUM(CASE WHEN i.estado = 'pendiente' THEN i.peso_seco ELSE 0 END), 0) as total_qq_seco_deposito
|
||||
|
||||
FROM vista_detalle_ingresos i
|
||||
LEFT JOIN clientes c ON i.cliente_id = c.id
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (i.estado != 'anulado' AND i.fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR i.created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR i.created_at <= {{fecha_hasta}})
|
||||
AND (CARDINALITY({{cliente_ids}}) = 0 OR i.cliente_id = ANY({{cliente_ids}}))
|
||||
AND (CARDINALITY({{tipos}}) = 0 OR i.tipo = ANY({{tipos}}))
|
||||
AND (CARDINALITY({{estados}}) = 0 OR i.estado = ANY({{estados}}))
|
||||
AND (CARDINALITY({{ubicaciones}}) = 0 OR c.ubicacion = ANY({{ubicaciones}}))
|
||||
AND (CARDINALITY({{calidades}}) = 0 OR i.calidad = ANY({{calidades}}));
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 2: `informe_totales_monetarios`
|
||||
**Reutiliza:** `panorama_totales_monetarios`
|
||||
**Componente:** `TotalesMonetarios.vue`
|
||||
|
||||
*(SQL idéntico a Query 3 de panorama, pero con los filtros adicionales como en Query 1)*
|
||||
|
||||
---
|
||||
|
||||
### Query 3: `informe_totales_verde`
|
||||
**Reutiliza:** `panorama_totales_verde`
|
||||
**Componente:** `TotalesVerde.vue`
|
||||
|
||||
*(SQL idéntico a Query 4 de panorama, pero con los filtros adicionales)*
|
||||
|
||||
---
|
||||
|
||||
## 📊 QUERIES NUEVAS PARA INFORME INGRESOS
|
||||
|
||||
### Query 4: `informe_lista_ingresos`
|
||||
**Componente:** `IngresosVistaTablaIngresos.vue`
|
||||
**Devuelve:** Múltiples filas (todos los ingresos filtrados con info del cliente)
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
i.id,
|
||||
i.cliente_id,
|
||||
i.created_at,
|
||||
i.fecha,
|
||||
i.tipo,
|
||||
i.estado,
|
||||
i.peso_neto,
|
||||
i.peso_seco,
|
||||
i.precio,
|
||||
i.calidad,
|
||||
i.observaciones,
|
||||
i.fecha_anulado,
|
||||
|
||||
-- Calcular total a pagar
|
||||
CASE
|
||||
WHEN i.tipo IN ('verde', 'uva') THEN i.precio * i.peso_neto
|
||||
WHEN i.tipo IN ('oreado', 'mojado') THEN (i.precio / 2) * i.peso_seco
|
||||
ELSE 0
|
||||
END as total_a_pagar,
|
||||
|
||||
-- Datos del cliente (para mostrar en la tabla)
|
||||
c.name as cliente_nombre,
|
||||
c.cedula as cliente_cedula,
|
||||
c.ubicacion as cliente_ubicacion,
|
||||
c.telefono as cliente_telefono
|
||||
|
||||
FROM vista_detalle_ingresos i
|
||||
LEFT JOIN clientes c ON i.cliente_id = c.id
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (i.estado != 'anulado' AND i.fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR i.created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR i.created_at <= {{fecha_hasta}})
|
||||
AND (CARDINALITY({{cliente_ids}}) = 0 OR i.cliente_id = ANY({{cliente_ids}}))
|
||||
AND (CARDINALITY({{tipos}}) = 0 OR i.tipo = ANY({{tipos}}))
|
||||
AND (CARDINALITY({{estados}}) = 0 OR i.estado = ANY({{estados}}))
|
||||
AND (CARDINALITY({{ubicaciones}}) = 0 OR c.ubicacion = ANY({{ubicaciones}}))
|
||||
AND (CARDINALITY({{calidades}}) = 0 OR i.calidad = ANY({{calidades}}))
|
||||
ORDER BY i.created_at DESC, i.id DESC
|
||||
LIMIT 1000;
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
[
|
||||
{
|
||||
"id": 1234,
|
||||
"cliente_id": 42,
|
||||
"created_at": "2025-01-15T10:30:00Z",
|
||||
"fecha": "2025-01-15",
|
||||
"tipo": "uva",
|
||||
"estado": "pagado",
|
||||
"peso_neto": 2550.00,
|
||||
"peso_seco": 25.50,
|
||||
"precio": 4.50,
|
||||
"calidad": "primera",
|
||||
"observaciones": null,
|
||||
"fecha_anulado": null,
|
||||
"total_a_pagar": 11475.00,
|
||||
"cliente_nombre": "Juan Pérez",
|
||||
"cliente_cedula": "0801199012345",
|
||||
"cliente_ubicacion": "El Rosario",
|
||||
"cliente_telefono": "99887766"
|
||||
}
|
||||
]
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 5: `informe_lista_clientes_con_totales`
|
||||
**Componente:** `ClientesVistaTablaClientes.vue`, `IngresosTopClientes.vue`
|
||||
**Devuelve:** Múltiples filas (un cliente por fila con sus totales agregados)
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
c.id as cliente_id,
|
||||
c.name as cliente_nombre,
|
||||
c.cedula as cliente_cedula,
|
||||
c.ubicacion as cliente_ubicacion,
|
||||
c.telefono as cliente_telefono,
|
||||
c.avatar_url as cliente_avatar,
|
||||
c.grupo_estudio as cliente_grupo_estudio,
|
||||
|
||||
-- === TOTALES POR CLIENTE ===
|
||||
|
||||
-- Contadores
|
||||
COUNT(i.id) as num_ingresos,
|
||||
|
||||
-- Totales de peso
|
||||
COALESCE(SUM(i.peso_seco), 0) as total_qq_seco,
|
||||
COALESCE(SUM(i.peso_neto), 0) as total_lb_neto,
|
||||
|
||||
-- Totales por tipo
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'uva' THEN i.peso_seco ELSE 0 END), 0) as total_qq_uva,
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'mojado' THEN i.peso_seco ELSE 0 END), 0) as total_qq_mojado,
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'oreado' THEN i.peso_seco ELSE 0 END), 0) as total_qq_oreado,
|
||||
COALESCE(SUM(CASE WHEN i.tipo = 'verde' THEN i.peso_neto ELSE 0 END), 0) as total_lb_verde,
|
||||
|
||||
-- Total pagado (inversión)
|
||||
COALESCE(SUM(
|
||||
CASE
|
||||
WHEN i.estado = 'pagado' AND i.tipo IN ('verde', 'uva') THEN i.precio * i.peso_neto
|
||||
WHEN i.estado = 'pagado' AND i.tipo IN ('oreado', 'mojado') THEN (i.precio / 2) * i.peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
), 0) as total_pagado,
|
||||
|
||||
-- Total pendiente de pagar
|
||||
COALESCE(SUM(
|
||||
CASE
|
||||
WHEN i.estado = 'pendiente' AND i.tipo IN ('verde', 'uva') THEN i.precio * i.peso_neto
|
||||
WHEN i.estado = 'pendiente' AND i.tipo IN ('oreado', 'mojado') THEN (i.precio / 2) * i.peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
), 0) as total_pendiente,
|
||||
|
||||
-- Precio promedio ponderado por qq
|
||||
CASE
|
||||
WHEN SUM(CASE WHEN i.estado = 'pagado' THEN i.peso_seco ELSE 0 END) > 0
|
||||
THEN SUM(
|
||||
CASE
|
||||
WHEN i.estado = 'pagado' AND i.tipo IN ('verde', 'uva') THEN i.precio * i.peso_neto
|
||||
WHEN i.estado = 'pagado' AND i.tipo IN ('oreado', 'mojado') THEN (i.precio / 2) * i.peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
) / SUM(CASE WHEN i.estado = 'pagado' THEN i.peso_seco ELSE 0 END)
|
||||
ELSE 0
|
||||
END as precio_promedio_qq,
|
||||
|
||||
-- Fecha primer ingreso
|
||||
MIN(i.created_at) as primer_ingreso,
|
||||
|
||||
-- Fecha último ingreso
|
||||
MAX(i.created_at) as ultimo_ingreso
|
||||
|
||||
FROM clientes c
|
||||
LEFT JOIN vista_detalle_ingresos i ON c.id = i.cliente_id
|
||||
AND ({{incluir_anulados}} OR (i.estado != 'anulado' AND i.fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR i.created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR i.created_at <= {{fecha_hasta}})
|
||||
AND (CARDINALITY({{tipos}}) = 0 OR i.tipo = ANY({{tipos}}))
|
||||
AND (CARDINALITY({{estados}}) = 0 OR i.estado = ANY({{estados}}))
|
||||
AND (CARDINALITY({{calidades}}) = 0 OR i.calidad = ANY({{calidades}}))
|
||||
|
||||
WHERE
|
||||
-- Filtro de clientes seleccionados
|
||||
(CARDINALITY({{cliente_ids}}) = 0 OR c.id = ANY({{cliente_ids}}))
|
||||
|
||||
-- Filtro de ubicaciones de clientes
|
||||
AND (CARDINALITY({{ubicaciones}}) = 0 OR c.ubicacion = ANY({{ubicaciones}}))
|
||||
|
||||
GROUP BY c.id, c.name, c.cedula, c.ubicacion, c.telefono, c.avatar_url, c.grupo_estudio
|
||||
HAVING COUNT(i.id) > 0
|
||||
ORDER BY total_pagado DESC;
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
[
|
||||
{
|
||||
"cliente_id": 42,
|
||||
"cliente_nombre": "Juan Pérez",
|
||||
"cliente_cedula": "0801199012345",
|
||||
"cliente_ubicacion": "El Rosario",
|
||||
"cliente_telefono": "99887766",
|
||||
"cliente_avatar": null,
|
||||
"cliente_grupo_estudio": null,
|
||||
"num_ingresos": 45,
|
||||
"total_qq_seco": 450.00,
|
||||
"total_lb_neto": 45000.00,
|
||||
"total_qq_uva": 200.00,
|
||||
"total_qq_mojado": 150.00,
|
||||
"total_qq_oreado": 100.00,
|
||||
"total_lb_verde": 5000.00,
|
||||
"total_pagado": 1234567.89,
|
||||
"total_pendiente": 50000.00,
|
||||
"precio_promedio_qq": 2743.71,
|
||||
"primer_ingreso": "2024-11-01T08:00:00Z",
|
||||
"ultimo_ingreso": "2025-01-15T14:30:00Z"
|
||||
}
|
||||
]
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 6: `informe_serie_temporal_acumulada`
|
||||
**Componente:** `GraficaAcumuladoresUva.vue`, `GraficaDinamicaPagadoDeposito.vue`
|
||||
**Devuelve:** Múltiples filas (agrupadas por fecha y tipo, con acumulación)
|
||||
**Parámetros adicionales:** `granularidad` (valores: 'dia', 'semana', 'mes')
|
||||
|
||||
```sql
|
||||
WITH datos_agrupados AS (
|
||||
SELECT
|
||||
-- Agrupar por granularidad
|
||||
CASE
|
||||
WHEN {{granularidad}} = 'dia' THEN DATE(i.created_at)
|
||||
WHEN {{granularidad}} = 'semana' THEN DATE_TRUNC('week', i.created_at)::date
|
||||
WHEN {{granularidad}} = 'mes' THEN DATE_TRUNC('month', i.created_at)::date
|
||||
ELSE DATE(i.created_at)
|
||||
END as fecha_grupo,
|
||||
|
||||
i.tipo,
|
||||
i.estado,
|
||||
|
||||
-- Totales del período
|
||||
SUM(i.peso_seco) as peso_seco_periodo,
|
||||
SUM(i.peso_neto) as peso_neto_periodo,
|
||||
SUM(
|
||||
CASE
|
||||
WHEN i.tipo IN ('verde', 'uva') THEN i.precio * i.peso_neto
|
||||
WHEN i.tipo IN ('oreado', 'mojado') THEN (i.precio / 2) * i.peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
) as inversion_periodo,
|
||||
COUNT(*) as num_ingresos_periodo
|
||||
|
||||
FROM vista_detalle_ingresos i
|
||||
LEFT JOIN clientes c ON i.cliente_id = c.id
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (i.estado != 'anulado' AND i.fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR i.created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR i.created_at <= {{fecha_hasta}})
|
||||
AND (CARDINALITY({{cliente_ids}}) = 0 OR i.cliente_id = ANY({{cliente_ids}}))
|
||||
AND (CARDINALITY({{tipos}}) = 0 OR i.tipo = ANY({{tipos}}))
|
||||
AND (CARDINALITY({{estados}}) = 0 OR i.estado = ANY({{estados}}))
|
||||
AND (CARDINALITY({{ubicaciones}}) = 0 OR c.ubicacion = ANY({{ubicaciones}}))
|
||||
AND (CARDINALITY({{calidades}}) = 0 OR i.calidad = ANY({{calidades}}))
|
||||
GROUP BY fecha_grupo, i.tipo, i.estado
|
||||
)
|
||||
SELECT
|
||||
fecha_grupo,
|
||||
tipo,
|
||||
estado,
|
||||
peso_seco_periodo,
|
||||
peso_neto_periodo,
|
||||
inversion_periodo,
|
||||
num_ingresos_periodo,
|
||||
|
||||
-- Acumulados hasta esta fecha (por tipo y estado)
|
||||
SUM(peso_seco_periodo) OVER (
|
||||
PARTITION BY tipo, estado
|
||||
ORDER BY fecha_grupo
|
||||
ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW
|
||||
) as peso_seco_acumulado,
|
||||
|
||||
SUM(peso_neto_periodo) OVER (
|
||||
PARTITION BY tipo, estado
|
||||
ORDER BY fecha_grupo
|
||||
ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW
|
||||
) as peso_neto_acumulado,
|
||||
|
||||
SUM(inversion_periodo) OVER (
|
||||
PARTITION BY tipo, estado
|
||||
ORDER BY fecha_grupo
|
||||
ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW
|
||||
) as inversion_acumulada
|
||||
|
||||
FROM datos_agrupados
|
||||
ORDER BY fecha_grupo, tipo, estado;
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
[
|
||||
{
|
||||
"fecha_grupo": "2025-01-15",
|
||||
"tipo": "uva",
|
||||
"estado": "pagado",
|
||||
"peso_seco_periodo": 25.50,
|
||||
"peso_neto_periodo": 2550.00,
|
||||
"inversion_periodo": 11475.00,
|
||||
"num_ingresos_periodo": 5,
|
||||
"peso_seco_acumulado": 450.00,
|
||||
"peso_neto_acumulado": 45000.00,
|
||||
"inversion_acumulada": 202500.00
|
||||
}
|
||||
]
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 7: `informe_opciones_filtros`
|
||||
**Propósito:** Obtener listas de valores únicos para los filtros dinámicos
|
||||
**Devuelve:** 1 objeto con arrays de opciones
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
-- Ubicaciones únicas de clientes que tienen ingresos
|
||||
(
|
||||
SELECT JSON_AGG(DISTINCT c.ubicacion ORDER BY c.ubicacion)
|
||||
FROM clientes c
|
||||
WHERE c.ubicacion IS NOT NULL
|
||||
) as ubicaciones_disponibles,
|
||||
|
||||
-- Calidades únicas de ingresos
|
||||
(
|
||||
SELECT JSON_AGG(DISTINCT i.calidad ORDER BY i.calidad)
|
||||
FROM vista_detalle_ingresos i
|
||||
WHERE i.calidad IS NOT NULL
|
||||
) as calidades_disponibles,
|
||||
|
||||
-- Tipos de café (hardcoded pero por consistencia)
|
||||
ARRAY['uva', 'mojado', 'oreado', 'verde']::text[] as tipos_cafe,
|
||||
|
||||
-- Estados (hardcoded)
|
||||
ARRAY['pagado', 'pendiente']::text[] as estados_disponibles;
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
{
|
||||
"ubicaciones_disponibles": ["El Rosario", "Las Flores", "Belén", "Santa Rosa"],
|
||||
"calidades_disponibles": ["primera", "segunda", "tercera", "especial"],
|
||||
"tipos_cafe": ["uva", "mojado", "oreado", "verde"],
|
||||
"estados_disponibles": ["pagado", "pendiente"]
|
||||
}
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 8: `informe_contadores_filtros`
|
||||
**Propósito:** Para el footer que muestra "X ingresos filtrados de Y totales"
|
||||
**Devuelve:** 1 fila con contadores
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
-- Total ingresos (sin filtros)
|
||||
(SELECT COUNT(*) FROM vista_detalle_ingresos) as total_ingresos,
|
||||
|
||||
-- Ingresos filtrados
|
||||
(
|
||||
SELECT COUNT(*)
|
||||
FROM vista_detalle_ingresos i
|
||||
LEFT JOIN clientes c ON i.cliente_id = c.id
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (i.estado != 'anulado' AND i.fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR i.created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR i.created_at <= {{fecha_hasta}})
|
||||
AND (CARDINALITY({{cliente_ids}}) = 0 OR i.cliente_id = ANY({{cliente_ids}}))
|
||||
AND (CARDINALITY({{tipos}}) = 0 OR i.tipo = ANY({{tipos}}))
|
||||
AND (CARDINALITY({{estados}}) = 0 OR i.estado = ANY({{estados}}))
|
||||
AND (CARDINALITY({{ubicaciones}}) = 0 OR c.ubicacion = ANY({{ubicaciones}}))
|
||||
AND (CARDINALITY({{calidades}}) = 0 OR i.calidad = ANY({{calidades}}))
|
||||
) as ingresos_filtrados,
|
||||
|
||||
-- Total clientes (sin filtros)
|
||||
(SELECT COUNT(*) FROM clientes) as total_clientes,
|
||||
|
||||
-- Clientes con ingresos filtrados
|
||||
(
|
||||
SELECT COUNT(DISTINCT i.cliente_id)
|
||||
FROM vista_detalle_ingresos i
|
||||
LEFT JOIN clientes c ON i.cliente_id = c.id
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (i.estado != 'anulado' AND i.fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR i.created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR i.created_at <= {{fecha_hasta}})
|
||||
AND (CARDINALITY({{cliente_ids}}) = 0 OR i.cliente_id = ANY({{cliente_ids}}))
|
||||
AND (CARDINALITY({{tipos}}) = 0 OR i.tipo = ANY({{tipos}}))
|
||||
AND (CARDINALITY({{estados}}) = 0 OR i.estado = ANY({{estados}}))
|
||||
AND (CARDINALITY({{ubicaciones}}) = 0 OR c.ubicacion = ANY({{ubicaciones}}))
|
||||
AND (CARDINALITY({{calidades}}) = 0 OR i.calidad = ANY({{calidades}}))
|
||||
) as clientes_con_ingresos_filtrados;
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
{
|
||||
"total_ingresos": 1250,
|
||||
"ingresos_filtrados": 230,
|
||||
"total_clientes": 85,
|
||||
"clientes_con_ingresos_filtrados": 42
|
||||
}
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 📊 RESUMEN DE QUERIES
|
||||
|
||||
| # | Nombre | Componente | Tipo de respuesta | Reutiliza? |
|
||||
|---|--------|------------|-------------------|------------|
|
||||
| 1 | `informe_totales_ingreso_compra` | TotalesIngresoCompra | 1 fila | ✅ Panorama Q2 |
|
||||
| 2 | `informe_totales_monetarios` | TotalesMonetarios | 1 fila | ✅ Panorama Q3 |
|
||||
| 3 | `informe_totales_verde` | TotalesVerde | 1 fila | ✅ Panorama Q4 |
|
||||
| 4 | `informe_lista_ingresos` | VistaTablaIngresos | Múltiples filas | ❌ Nueva |
|
||||
| 5 | `informe_lista_clientes_con_totales` | VistaTablaClientes, TopClientes | Múltiples filas | ❌ Nueva |
|
||||
| 6 | `informe_serie_temporal_acumulada` | Gráficas acumuladas | Múltiples filas | ❌ Nueva |
|
||||
| 7 | `informe_opciones_filtros` | Selectores de filtros | 1 objeto JSON | ❌ Nueva |
|
||||
| 8 | `informe_contadores_filtros` | Footer de filtros | 1 fila | ❌ Nueva |
|
||||
|
||||
**Total: 8 Queries (3 reutilizadas + 5 nuevas)**
|
||||
|
||||
---
|
||||
|
||||
## 🔧 CONFIGURACIÓN DE PARÁMETROS EN METABASE
|
||||
|
||||
### Parámetros Simples (igual que panorama):
|
||||
- `fecha_desde`: Date, opcional
|
||||
- `fecha_hasta`: Date, opcional
|
||||
- `incluir_anulados`: Boolean, default `false`
|
||||
|
||||
### Parámetros de Array (NUEVO):
|
||||
|
||||
#### `cliente_ids`
|
||||
- **Tipo:** Text (se parsea como array)
|
||||
- **Widget Type:** Text
|
||||
- **Default:** `[]`
|
||||
- **Required:** No
|
||||
- **Variable Name:** `cliente_ids`
|
||||
- **SQL Type:** `INTEGER[]`
|
||||
|
||||
#### `tipos`
|
||||
- **Tipo:** Text
|
||||
- **Widget Type:** Text o Dropdown (multi-select)
|
||||
- **Default:** `[]`
|
||||
- **Required:** No
|
||||
- **Options:** `['uva', 'mojado', 'oreado', 'verde']`
|
||||
- **SQL Type:** `TEXT[]`
|
||||
|
||||
#### `estados`
|
||||
- **Tipo:** Text
|
||||
- **Widget Type:** Text o Dropdown (multi-select)
|
||||
- **Default:** `[]`
|
||||
- **Required:** No
|
||||
- **Options:** `['pagado', 'pendiente']`
|
||||
- **SQL Type:** `TEXT[]`
|
||||
|
||||
#### `ubicaciones`
|
||||
- **Tipo:** Text
|
||||
- **Widget Type:** Text
|
||||
- **Default:** `[]`
|
||||
- **Required:** No
|
||||
- **SQL Type:** `TEXT[]`
|
||||
|
||||
#### `calidades`
|
||||
- **Tipo:** Text
|
||||
- **Widget Type:** Text
|
||||
- **Default:** `[]`
|
||||
- **Required:** No
|
||||
- **SQL Type:** `TEXT[]`
|
||||
|
||||
#### `granularidad` (solo Query 6)
|
||||
- **Tipo:** Text
|
||||
- **Widget Type:** Dropdown
|
||||
- **Default:** `'dia'`
|
||||
- **Required:** Yes
|
||||
- **Options:** `['dia', 'semana', 'mes']`
|
||||
|
||||
---
|
||||
|
||||
## 🚀 INTEGRACIÓN EN VUE
|
||||
|
||||
```typescript
|
||||
// informe-ingresos.vue
|
||||
|
||||
const filters = ref({
|
||||
fecha_desde: null,
|
||||
fecha_hasta: null,
|
||||
incluir_anulados: false,
|
||||
cliente_ids: [], // Array de números
|
||||
tipos: [], // Array de strings
|
||||
estados: [], // Array de strings
|
||||
ubicaciones: [], // Array de strings
|
||||
calidades: [], // Array de strings
|
||||
granularidad: 'dia' // Solo para Query 6
|
||||
})
|
||||
|
||||
async function loadAllData() {
|
||||
const [
|
||||
totalesIngresoCompra,
|
||||
totalesMonetarios,
|
||||
totalesVerde,
|
||||
listaIngresos,
|
||||
listaClientes,
|
||||
serieAcumulada,
|
||||
opcionesFiltros,
|
||||
contadores
|
||||
] = await Promise.all([
|
||||
$fetch('/api/metabase/question/101', { query: filters.value }),
|
||||
$fetch('/api/metabase/question/102', { query: filters.value }),
|
||||
$fetch('/api/metabase/question/103', { query: filters.value }),
|
||||
$fetch('/api/metabase/question/104', { query: filters.value }),
|
||||
$fetch('/api/metabase/question/105', { query: filters.value }),
|
||||
$fetch('/api/metabase/question/106', { query: filters.value }),
|
||||
$fetch('/api/metabase/question/107'), // Sin filtros, siempre trae todo
|
||||
$fetch('/api/metabase/question/108', { query: filters.value })
|
||||
])
|
||||
|
||||
// Asignar directamente
|
||||
metrics.value = {
|
||||
totalesIngresoCompra: totalesIngresoCompra.data.rows[0],
|
||||
totalesMonetarios: totalesMonetarios.data.rows[0],
|
||||
totalesVerde: totalesVerde.data.rows[0],
|
||||
ingresos: listaIngresos.data.rows,
|
||||
clientes: listaClientes.data.rows,
|
||||
serieAcumulada: serieAcumulada.data.rows,
|
||||
opcionesFiltros: opcionesFiltros.data.rows[0],
|
||||
contadores: contadores.data.rows[0]
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## ⚠️ NOTAS IMPORTANTES
|
||||
|
||||
### 1. Soporte de Arrays en Metabase
|
||||
|
||||
Metabase no tiene soporte nativo para arrays en parámetros. Hay dos opciones:
|
||||
|
||||
**Opción A: Usar PostgreSQL arrays con CARDINALITY**
|
||||
```sql
|
||||
WHERE (CARDINALITY({{cliente_ids}}) = 0 OR i.cliente_id = ANY({{cliente_ids}}))
|
||||
```
|
||||
El parámetro se pasa como: `{1,2,3}` o `ARRAY[1,2,3]`
|
||||
|
||||
**Opción B: Usar múltiples OR con unnest**
|
||||
```sql
|
||||
WHERE ({{cliente_ids}} = '' OR i.cliente_id IN (SELECT unnest(string_to_array({{cliente_ids}}, ','))))
|
||||
```
|
||||
El parámetro se pasa como string CSV: `"1,2,3"`
|
||||
|
||||
**Recomendación:** Usar Opción A si Metabase/PostgreSQL lo soporta, caso contrario usar Opción B.
|
||||
|
||||
### 2. Performance
|
||||
|
||||
- Query 4 (lista de ingresos) tiene LIMIT 1000 para evitar cargas masivas
|
||||
- Query 5 (clientes con totales) puede ser lenta si hay muchos ingresos
|
||||
- Query 6 (serie acumulada) usa window functions, puede requerir índices en `created_at`
|
||||
|
||||
### 3. Índices Recomendados
|
||||
|
||||
```sql
|
||||
CREATE INDEX idx_ingresos_created_at ON vista_detalle_ingresos(created_at);
|
||||
CREATE INDEX idx_ingresos_cliente_id ON vista_detalle_ingresos(cliente_id);
|
||||
CREATE INDEX idx_ingresos_tipo ON vista_detalle_ingresos(tipo);
|
||||
CREATE INDEX idx_ingresos_estado ON vista_detalle_ingresos(estado);
|
||||
CREATE INDEX idx_clientes_ubicacion ON clientes(ubicacion);
|
||||
```
|
||||
|
||||
### 4. Diferencias con Panorama
|
||||
|
||||
- Panorama usa filtros simples (un valor por parámetro)
|
||||
- Informe usa filtros múltiples (arrays de valores)
|
||||
- Informe tiene query de lista completa de ingresos (para tabla)
|
||||
- Informe tiene agregación por cliente (para ranking y tabla de clientes)
|
||||
|
||||
---
|
||||
|
||||
## 📝 PRÓXIMOS PASOS
|
||||
|
||||
1. ✅ Crear estas 8 queries en Metabase
|
||||
2. ✅ Configurar parámetros (especialmente los de tipo array)
|
||||
3. ✅ Probar queries con diferentes combinaciones de filtros
|
||||
4. ✅ Obtener los Question IDs
|
||||
5. ✅ Adaptar `informe-ingresos.vue` para usar las nuevas queries
|
||||
6. ✅ Eliminar composables de métricas (reutilizar estructura de panorama)
|
||||
7. ✅ Adaptar componentes para recibir props directamente
|
||||
8. ✅ Probar performance con datasets grandes
|
||||
|
||||
---
|
||||
|
||||
**Documento creado:** 2025-10-14
|
||||
**Autor:** Claude Code
|
||||
**Proyecto:** Analítica Núcleo - Informe Ingresos
|
||||
734
METABASE_QUERIES_PANORAMA.md
Normal file
734
METABASE_QUERIES_PANORAMA.md
Normal file
@@ -0,0 +1,734 @@
|
||||
# 📊 QUERIES DE METABASE PARA PANORAMA FACTURADOR
|
||||
|
||||
## 🎯 FILOSOFÍA
|
||||
|
||||
**Metabase calcula TODO. Vue solo renderiza.**
|
||||
|
||||
Cada query devuelve **exactamente los valores que un componente necesita mostrar**, ya calculados y listos para usar. Sin composables de métricas, sin cálculos en el frontend.
|
||||
|
||||
---
|
||||
|
||||
## 📋 PARÁMETROS GLOBALES
|
||||
|
||||
Todas las queries aceptan estos parámetros:
|
||||
|
||||
| Parámetro | Tipo | Default | Descripción |
|
||||
|-----------|------|---------|-------------|
|
||||
| `fecha_desde` | Date | `null` | Fecha inicio (filtra por `created_at`) |
|
||||
| `fecha_hasta` | Date | `null` | Fecha fin (filtra por `created_at`) |
|
||||
| `incluir_anulados` | Boolean | `false` | Si `false`, excluye `estado='anulado'` o `fecha_anulado IS NOT NULL` |
|
||||
|
||||
**Condición SQL estándar:**
|
||||
```sql
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}})
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 🎨 QUERIES POR COMPONENTE
|
||||
|
||||
### Query 1: `panorama_totales_financieros_principales`
|
||||
**Componente:** Card principal de "Totales Financieros"
|
||||
**Devuelve:** 1 fila con 3 valores
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
-- Total Invertido en Café (pagado: uva + mojado + oreado)
|
||||
COALESCE(SUM(
|
||||
CASE
|
||||
WHEN estado = 'pagado' AND tipo IN ('verde', 'uva')
|
||||
THEN precio * peso_neto
|
||||
WHEN estado = 'pagado' AND tipo IN ('oreado', 'mojado')
|
||||
THEN (precio / 2) * peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
), 0) as total_invertido_cafe,
|
||||
|
||||
-- Total Rechazos (traer de otra tabla)
|
||||
COALESCE((
|
||||
SELECT SUM(total_cobrado)
|
||||
FROM rechazos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}})
|
||||
), 0) as total_rechazos,
|
||||
|
||||
-- Balance Neto (invertido - rechazos)
|
||||
COALESCE(SUM(
|
||||
CASE
|
||||
WHEN estado = 'pagado' AND tipo IN ('verde', 'uva')
|
||||
THEN precio * peso_neto
|
||||
WHEN estado = 'pagado' AND tipo IN ('oreado', 'mojado')
|
||||
THEN (precio / 2) * peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
), 0) - COALESCE((
|
||||
SELECT SUM(total_cobrado)
|
||||
FROM rechazos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}})
|
||||
), 0) as balance_neto
|
||||
|
||||
FROM vista_detalle_ingresos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}});
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
{
|
||||
"total_invertido_cafe": 4567890.12,
|
||||
"total_rechazos": 44750.00,
|
||||
"balance_neto": 4523140.12
|
||||
}
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 2: `panorama_totales_ingreso_compra`
|
||||
**Componente:** `TotalesIngresoCompra.vue`
|
||||
**Devuelve:** 1 fila con TODOS los valores del componente
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
-- === TOTALES GENERALES (Pagado + Pendiente) ===
|
||||
|
||||
-- Uva
|
||||
COALESCE(SUM(CASE WHEN tipo = 'uva' THEN peso_neto ELSE 0 END), 0) as total_lb_uva_ingresada,
|
||||
COALESCE(SUM(CASE WHEN tipo = 'uva' THEN peso_seco ELSE 0 END), 0) as total_qq_seco_uva_ingresado,
|
||||
|
||||
-- Mojado
|
||||
COALESCE(SUM(CASE WHEN tipo = 'mojado' THEN peso_seco ELSE 0 END), 0) as total_qq_seco_mojado_ingresado,
|
||||
|
||||
-- Oreado
|
||||
COALESCE(SUM(CASE WHEN tipo = 'oreado' THEN peso_seco ELSE 0 END), 0) as total_qq_seco_oreado_ingresado,
|
||||
|
||||
-- Total general
|
||||
COALESCE(SUM(peso_seco), 0) as total_qq_seco_ingresado,
|
||||
|
||||
-- === SOLO PAGADOS ===
|
||||
|
||||
-- Uva pagada
|
||||
COALESCE(SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN peso_neto ELSE 0 END), 0) as total_lb_uva_pagada,
|
||||
COALESCE(SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN peso_seco ELSE 0 END), 0) as total_qq_seco_uva_pagado,
|
||||
|
||||
-- Mojado pagado
|
||||
COALESCE(SUM(CASE WHEN tipo = 'mojado' AND estado = 'pagado' THEN peso_seco ELSE 0 END), 0) as total_qq_seco_mojado_pagado,
|
||||
|
||||
-- Oreado pagado
|
||||
COALESCE(SUM(CASE WHEN tipo = 'oreado' AND estado = 'pagado' THEN peso_seco ELSE 0 END), 0) as total_qq_seco_oreado_pagado,
|
||||
|
||||
-- Total pagado
|
||||
COALESCE(SUM(CASE WHEN estado = 'pagado' THEN peso_seco ELSE 0 END), 0) as total_qq_seco_comprado,
|
||||
|
||||
-- === INVENTARIO EN DEPÓSITO (Pendiente) ===
|
||||
|
||||
-- Uva en depósito
|
||||
COALESCE(SUM(CASE WHEN tipo = 'uva' AND estado = 'pendiente' THEN peso_neto ELSE 0 END), 0) as total_lb_uva_deposito,
|
||||
|
||||
-- Mojado en depósito
|
||||
COALESCE(SUM(CASE WHEN tipo = 'mojado' AND estado = 'pendiente' THEN peso_seco ELSE 0 END), 0) as total_qq_mojado_deposito,
|
||||
|
||||
-- Oreado en depósito
|
||||
COALESCE(SUM(CASE WHEN tipo = 'oreado' AND estado = 'pendiente' THEN peso_seco ELSE 0 END), 0) as total_qq_oreado_deposito,
|
||||
|
||||
-- Total en depósito
|
||||
COALESCE(SUM(CASE WHEN estado = 'pendiente' THEN peso_seco ELSE 0 END), 0) as total_qq_seco_deposito
|
||||
|
||||
FROM vista_detalle_ingresos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}});
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
{
|
||||
"total_lb_uva_ingresada": 123456.78,
|
||||
"total_qq_seco_uva_ingresado": 246.91,
|
||||
"total_qq_seco_mojado_ingresado": 500.00,
|
||||
"total_qq_seco_oreado_ingresado": 300.00,
|
||||
"total_qq_seco_ingresado": 1046.91,
|
||||
"total_lb_uva_pagada": 100000.00,
|
||||
"total_qq_seco_uva_pagado": 200.00,
|
||||
"total_qq_seco_mojado_pagado": 450.00,
|
||||
"total_qq_seco_oreado_pagado": 250.00,
|
||||
"total_qq_seco_comprado": 900.00,
|
||||
"total_lb_uva_deposito": 23456.78,
|
||||
"total_qq_mojado_deposito": 50.00,
|
||||
"total_qq_oreado_deposito": 50.00,
|
||||
"total_qq_seco_deposito": 146.91
|
||||
}
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 3: `panorama_totales_monetarios`
|
||||
**Componente:** `TotalesMonetarios.vue`
|
||||
**Devuelve:** 1 fila con TODOS los valores
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
-- === INVERSIÓN HASTA LA FECHA ===
|
||||
|
||||
COALESCE(SUM(
|
||||
CASE
|
||||
WHEN estado = 'pagado' AND tipo = 'uva'
|
||||
THEN precio * peso_neto
|
||||
ELSE 0
|
||||
END
|
||||
), 0) as inversion_uva,
|
||||
|
||||
COALESCE(SUM(
|
||||
CASE
|
||||
WHEN estado = 'pagado' AND tipo = 'mojado'
|
||||
THEN (precio / 2) * peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
), 0) as inversion_mojado,
|
||||
|
||||
COALESCE(SUM(
|
||||
CASE
|
||||
WHEN estado = 'pagado' AND tipo = 'oreado'
|
||||
THEN (precio / 2) * peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
), 0) as inversion_oreado,
|
||||
|
||||
COALESCE(SUM(
|
||||
CASE
|
||||
WHEN estado = 'pagado' AND tipo IN ('uva', 'mojado', 'oreado')
|
||||
THEN
|
||||
CASE
|
||||
WHEN tipo = 'uva' THEN precio * peso_neto
|
||||
ELSE (precio / 2) * peso_seco
|
||||
END
|
||||
ELSE 0
|
||||
END
|
||||
), 0) as total_invertido,
|
||||
|
||||
-- === PRECIOS PROMEDIO PONDERADOS ===
|
||||
|
||||
-- Precio promedio uva por lb
|
||||
CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN peso_neto ELSE 0 END) > 0
|
||||
THEN SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN precio * peso_neto ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN peso_neto ELSE 0 END)
|
||||
ELSE 0
|
||||
END as precio_promedio_uva_por_lb,
|
||||
|
||||
-- Precio promedio uva por qq (lb * 500)
|
||||
CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN peso_neto ELSE 0 END) > 0
|
||||
THEN (SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN precio * peso_neto ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN peso_neto ELSE 0 END)) * 500
|
||||
ELSE 0
|
||||
END as precio_promedio_uva_por_qq,
|
||||
|
||||
-- Precio promedio mojado por qq
|
||||
CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'mojado' AND estado = 'pagado' THEN peso_seco ELSE 0 END) > 0
|
||||
THEN SUM(CASE WHEN tipo = 'mojado' AND estado = 'pagado' THEN (precio / 2) * peso_seco ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'mojado' AND estado = 'pagado' THEN peso_seco ELSE 0 END)
|
||||
ELSE 0
|
||||
END as precio_promedio_mojado_por_qq,
|
||||
|
||||
-- Precio promedio oreado por qq
|
||||
CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'oreado' AND estado = 'pagado' THEN peso_seco ELSE 0 END) > 0
|
||||
THEN SUM(CASE WHEN tipo = 'oreado' AND estado = 'pagado' THEN (precio / 2) * peso_seco ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'oreado' AND estado = 'pagado' THEN peso_seco ELSE 0 END)
|
||||
ELSE 0
|
||||
END as precio_promedio_oreado_por_qq,
|
||||
|
||||
-- Precio promedio global qq seco
|
||||
CASE
|
||||
WHEN SUM(CASE WHEN estado = 'pagado' THEN peso_seco ELSE 0 END) > 0
|
||||
THEN SUM(
|
||||
CASE
|
||||
WHEN estado = 'pagado' AND tipo IN ('verde', 'uva')
|
||||
THEN precio * peso_neto
|
||||
WHEN estado = 'pagado' AND tipo IN ('oreado', 'mojado')
|
||||
THEN (precio / 2) * peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
) / SUM(CASE WHEN estado = 'pagado' THEN peso_seco ELSE 0 END)
|
||||
ELSE 0
|
||||
END as precio_promedio_qq_seco,
|
||||
|
||||
-- === INVERSIÓN RESTANTE A REALIZAR ===
|
||||
|
||||
-- Inversión restante uva (precio_promedio_uva_lb * lb_deposito)
|
||||
(CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN peso_neto ELSE 0 END) > 0
|
||||
THEN SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN precio * peso_neto ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN peso_neto ELSE 0 END)
|
||||
ELSE 0
|
||||
END) * COALESCE(SUM(CASE WHEN tipo = 'uva' AND estado = 'pendiente' THEN peso_neto ELSE 0 END), 0) as inversion_restante_uva,
|
||||
|
||||
-- Inversión restante mojado (precio_promedio_mojado_qq * qq_deposito)
|
||||
(CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'mojado' AND estado = 'pagado' THEN peso_seco ELSE 0 END) > 0
|
||||
THEN SUM(CASE WHEN tipo = 'mojado' AND estado = 'pagado' THEN (precio / 2) * peso_seco ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'mojado' AND estado = 'pagado' THEN peso_seco ELSE 0 END)
|
||||
ELSE 0
|
||||
END) * COALESCE(SUM(CASE WHEN tipo = 'mojado' AND estado = 'pendiente' THEN peso_seco ELSE 0 END), 0) as inversion_restante_mojado,
|
||||
|
||||
-- Inversión restante oreado (precio_promedio_oreado_qq * qq_deposito)
|
||||
(CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'oreado' AND estado = 'pagado' THEN peso_seco ELSE 0 END) > 0
|
||||
THEN SUM(CASE WHEN tipo = 'oreado' AND estado = 'pagado' THEN (precio / 2) * peso_seco ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'oreado' AND estado = 'pagado' THEN peso_seco ELSE 0 END)
|
||||
ELSE 0
|
||||
END) * COALESCE(SUM(CASE WHEN tipo = 'oreado' AND estado = 'pendiente' THEN peso_seco ELSE 0 END), 0) as inversion_restante_oreado,
|
||||
|
||||
-- Inversión restante esperada (suma de las 3 anteriores)
|
||||
(CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN peso_neto ELSE 0 END) > 0
|
||||
THEN SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN precio * peso_neto ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'uva' AND estado = 'pagado' THEN peso_neto ELSE 0 END)
|
||||
ELSE 0
|
||||
END) * COALESCE(SUM(CASE WHEN tipo = 'uva' AND estado = 'pendiente' THEN peso_neto ELSE 0 END), 0) +
|
||||
(CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'mojado' AND estado = 'pagado' THEN peso_seco ELSE 0 END) > 0
|
||||
THEN SUM(CASE WHEN tipo = 'mojado' AND estado = 'pagado' THEN (precio / 2) * peso_seco ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'mojado' AND estado = 'pagado' THEN peso_seco ELSE 0 END)
|
||||
ELSE 0
|
||||
END) * COALESCE(SUM(CASE WHEN tipo = 'mojado' AND estado = 'pendiente' THEN peso_seco ELSE 0 END), 0) +
|
||||
(CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'oreado' AND estado = 'pagado' THEN peso_seco ELSE 0 END) > 0
|
||||
THEN SUM(CASE WHEN tipo = 'oreado' AND estado = 'pagado' THEN (precio / 2) * peso_seco ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'oreado' AND estado = 'pagado' THEN peso_seco ELSE 0 END)
|
||||
ELSE 0
|
||||
END) * COALESCE(SUM(CASE WHEN tipo = 'oreado' AND estado = 'pendiente' THEN peso_seco ELSE 0 END), 0) as inversion_restante_esperada
|
||||
|
||||
FROM vista_detalle_ingresos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}});
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
{
|
||||
"inversion_uva": 1234567.89,
|
||||
"inversion_mojado": 567890.12,
|
||||
"inversion_oreado": 345678.90,
|
||||
"total_invertido": 2148136.91,
|
||||
"precio_promedio_uva_por_lb": 4.50,
|
||||
"precio_promedio_uva_por_qq": 2250.00,
|
||||
"precio_promedio_mojado_por_qq": 2500.00,
|
||||
"precio_promedio_oreado_por_qq": 2300.00,
|
||||
"precio_promedio_qq_seco": 2400.00,
|
||||
"inversion_restante_uva": 105555.51,
|
||||
"inversion_restante_mojado": 125000.00,
|
||||
"inversion_restante_oreado": 115000.00,
|
||||
"inversion_restante_esperada": 345555.51
|
||||
}
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 4: `panorama_totales_verde`
|
||||
**Componente:** `TotalesVerde.vue`
|
||||
**Devuelve:** 1 fila con TODOS los valores
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
-- Total lb neto de verde (pagado + pendiente)
|
||||
COALESCE(SUM(CASE WHEN tipo = 'verde' THEN peso_neto ELSE 0 END), 0) as total_lb_neto_verde,
|
||||
|
||||
-- Precio promedio verde pagado (por lb)
|
||||
CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'verde' AND estado = 'pagado' THEN peso_neto ELSE 0 END) > 0
|
||||
THEN SUM(CASE WHEN tipo = 'verde' AND estado = 'pagado' THEN precio * peso_neto ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'verde' AND estado = 'pagado' THEN peso_neto ELSE 0 END)
|
||||
ELSE 0
|
||||
END as precio_promedio_verde_pagado,
|
||||
|
||||
-- Total lb neto verde en depósito (pendiente)
|
||||
COALESCE(SUM(CASE WHEN tipo = 'verde' AND estado = 'pendiente' THEN peso_neto ELSE 0 END), 0) as total_lb_neto_verde_deposito,
|
||||
|
||||
-- Inversión verde hasta la fecha (pagado)
|
||||
COALESCE(SUM(
|
||||
CASE
|
||||
WHEN tipo = 'verde' AND estado = 'pagado'
|
||||
THEN precio * peso_neto
|
||||
ELSE 0
|
||||
END
|
||||
), 0) as inversion_verde_hasta_fecha,
|
||||
|
||||
-- Inversión restante verde (precio_promedio * lb_deposito)
|
||||
(CASE
|
||||
WHEN SUM(CASE WHEN tipo = 'verde' AND estado = 'pagado' THEN peso_neto ELSE 0 END) > 0
|
||||
THEN SUM(CASE WHEN tipo = 'verde' AND estado = 'pagado' THEN precio * peso_neto ELSE 0 END) /
|
||||
SUM(CASE WHEN tipo = 'verde' AND estado = 'pagado' THEN peso_neto ELSE 0 END)
|
||||
ELSE 0
|
||||
END) * COALESCE(SUM(CASE WHEN tipo = 'verde' AND estado = 'pendiente' THEN peso_neto ELSE 0 END), 0) as inversion_restante_verde,
|
||||
|
||||
-- Total lb neto comprado verde (solo pagado)
|
||||
COALESCE(SUM(CASE WHEN tipo = 'verde' AND estado = 'pagado' THEN peso_neto ELSE 0 END), 0) as total_lb_neto_comprado_verde
|
||||
|
||||
FROM vista_detalle_ingresos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}});
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
{
|
||||
"total_lb_neto_verde": 50000.00,
|
||||
"precio_promedio_verde_pagado": 10.00,
|
||||
"total_lb_neto_verde_deposito": 5000.00,
|
||||
"inversion_verde_hasta_fecha": 450000.00,
|
||||
"inversion_restante_verde": 50000.00,
|
||||
"total_lb_neto_comprado_verde": 45000.00
|
||||
}
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 5: `panorama_secos_vendidos`
|
||||
**Componente:** `SecosVendidos.vue`
|
||||
**Devuelve:** 1 fila con TODOS los valores
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
-- Total qq seco por vender (en depósito)
|
||||
COALESCE(SUM(CASE WHEN estado = 'pendiente' THEN peso_seco ELSE 0 END), 0) as total_qq_seco_por_vender,
|
||||
|
||||
-- Precio de venta promedio por qq (TODO: necesita tabla de ventas, por ahora 0)
|
||||
0 as precio_venta_promedio_por_qq,
|
||||
|
||||
-- Precio de compra promedio por qq
|
||||
CASE
|
||||
WHEN SUM(CASE WHEN estado = 'pagado' THEN peso_seco ELSE 0 END) > 0
|
||||
THEN SUM(
|
||||
CASE
|
||||
WHEN estado = 'pagado' AND tipo IN ('verde', 'uva')
|
||||
THEN precio * peso_neto
|
||||
WHEN estado = 'pagado' AND tipo IN ('oreado', 'mojado')
|
||||
THEN (precio / 2) * peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
) / SUM(CASE WHEN estado = 'pagado' THEN peso_seco ELSE 0 END)
|
||||
ELSE 0
|
||||
END as precio_compra_promedio_por_qq,
|
||||
|
||||
-- Margen de ganancia por qq (venta - compra, por ahora será negativo)
|
||||
0 - CASE
|
||||
WHEN SUM(CASE WHEN estado = 'pagado' THEN peso_seco ELSE 0 END) > 0
|
||||
THEN SUM(
|
||||
CASE
|
||||
WHEN estado = 'pagado' AND tipo IN ('verde', 'uva')
|
||||
THEN precio * peso_neto
|
||||
WHEN estado = 'pagado' AND tipo IN ('oreado', 'mojado')
|
||||
THEN (precio / 2) * peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
) / SUM(CASE WHEN estado = 'pagado' THEN peso_seco ELSE 0 END)
|
||||
ELSE 0
|
||||
END as margen_ganancia_por_qq
|
||||
|
||||
FROM vista_detalle_ingresos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}});
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
{
|
||||
"total_qq_seco_por_vender": 146.91,
|
||||
"precio_venta_promedio_por_qq": 0,
|
||||
"precio_compra_promedio_por_qq": 2400.00,
|
||||
"margen_ganancia_por_qq": -2400.00
|
||||
}
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 6: `panorama_rechazos_subproductos`
|
||||
**Componente:** `RechazosSubproductos.vue`
|
||||
**Devuelve:** 6 filas (una por cada tipo de rechazo)
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
tipo,
|
||||
COUNT(*) as num_registros,
|
||||
COALESCE(SUM(cantidad), 0) as total_cantidad,
|
||||
COALESCE(SUM(total_cobrado), 0) as total_cobrado,
|
||||
|
||||
-- Precio promedio
|
||||
CASE
|
||||
WHEN SUM(cantidad) > 0 THEN SUM(total_cobrado) / SUM(cantidad)
|
||||
ELSE 0
|
||||
END as precio_promedio
|
||||
|
||||
FROM rechazos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}})
|
||||
GROUP BY tipo
|
||||
ORDER BY tipo;
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
[
|
||||
{ "tipo": "chibolita", "num_registros": 25, "total_cantidad": 1250.00, "total_cobrado": 12500.00, "precio_promedio": 10.00 },
|
||||
{ "tipo": "magalla", "num_registros": 10, "total_cantidad": 50.00, "total_cobrado": 2500.00, "precio_promedio": 50.00 },
|
||||
{ "tipo": "perico", "num_registros": 30, "total_cantidad": 1500.00, "total_cobrado": 15000.00, "precio_promedio": 10.00 },
|
||||
{ "tipo": "picadillo", "num_registros": 15, "total_cantidad": 75.00, "total_cobrado": 3750.00, "precio_promedio": 50.00 },
|
||||
{ "tipo": "pinta", "num_registros": 12, "total_cantidad": 600.00, "total_cobrado": 6000.00, "precio_promedio": 10.00 },
|
||||
{ "tipo": "vano", "num_registros": 20, "total_cantidad": 100.00, "total_cobrado": 5000.00, "precio_promedio": 50.00 }
|
||||
]
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 📈 QUERIES ADICIONALES (GRÁFICAS Y SERIES)
|
||||
|
||||
### Query 7: `panorama_serie_temporal_diaria`
|
||||
**Componente:** `GraficaSerieIngresos.vue`, `GraficaSerieInversion.vue`
|
||||
**Devuelve:** Múltiples filas (una por día/tipo/estado)
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
DATE(created_at) as fecha,
|
||||
tipo,
|
||||
estado,
|
||||
COUNT(*) as num_ingresos,
|
||||
COALESCE(SUM(peso_seco), 0) as total_peso_seco_dia,
|
||||
COALESCE(SUM(peso_neto), 0) as total_peso_neto_dia,
|
||||
COALESCE(SUM(
|
||||
CASE
|
||||
WHEN tipo IN ('verde', 'uva') THEN precio * peso_neto
|
||||
WHEN tipo IN ('oreado', 'mojado') THEN (precio / 2) * peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
), 0) as total_inversion_dia
|
||||
|
||||
FROM vista_detalle_ingresos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}})
|
||||
GROUP BY DATE(created_at), tipo, estado
|
||||
ORDER BY fecha, tipo, estado;
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 8: `panorama_top_clientes`
|
||||
**Componente:** `TopClientes.vue`
|
||||
**Devuelve:** Múltiples filas (top 20 clientes por inversión)
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
cliente_id,
|
||||
COUNT(*) as num_ingresos,
|
||||
COALESCE(SUM(peso_seco), 0) as total_qq_seco,
|
||||
COALESCE(SUM(peso_neto), 0) as total_lb_neto,
|
||||
COALESCE(SUM(
|
||||
CASE
|
||||
WHEN tipo IN ('verde', 'uva') THEN precio * peso_neto
|
||||
WHEN tipo IN ('oreado', 'mojado') THEN (precio / 2) * peso_seco
|
||||
ELSE 0
|
||||
END
|
||||
), 0) as total_pagado
|
||||
|
||||
FROM vista_detalle_ingresos
|
||||
WHERE
|
||||
estado = 'pagado'
|
||||
AND ({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}})
|
||||
GROUP BY cliente_id
|
||||
ORDER BY total_pagado DESC
|
||||
LIMIT 20;
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
### Query 9: `panorama_conteo_registros`
|
||||
**Propósito:** Para el footer que dice "Registros considerados: Ingresos X/Y · Rechazos A/B"
|
||||
**Devuelve:** 1 fila con conteos
|
||||
|
||||
```sql
|
||||
SELECT
|
||||
-- Ingresos filtrados
|
||||
(SELECT COUNT(*)
|
||||
FROM vista_detalle_ingresos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}})
|
||||
) as ingresos_filtrados,
|
||||
|
||||
-- Total ingresos (sin filtros de fecha/anulados)
|
||||
(SELECT COUNT(*) FROM vista_detalle_ingresos) as ingresos_total,
|
||||
|
||||
-- Rechazos filtrados
|
||||
(SELECT COUNT(*)
|
||||
FROM rechazos
|
||||
WHERE
|
||||
({{incluir_anulados}} OR (estado != 'anulado' AND fecha_anulado IS NULL))
|
||||
AND ({{fecha_desde}} IS NULL OR created_at >= {{fecha_desde}})
|
||||
AND ({{fecha_hasta}} IS NULL OR created_at <= {{fecha_hasta}})
|
||||
) as rechazos_filtrados,
|
||||
|
||||
-- Total rechazos
|
||||
(SELECT COUNT(*) FROM rechazos) as rechazos_total;
|
||||
```
|
||||
|
||||
**Respuesta esperada:**
|
||||
```json
|
||||
{
|
||||
"ingresos_filtrados": 230,
|
||||
"ingresos_total": 350,
|
||||
"rechazos_filtrados": 112,
|
||||
"rechazos_total": 150
|
||||
}
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 📊 RESUMEN DE QUERIES
|
||||
|
||||
| # | Nombre | Componente | Tipo de respuesta |
|
||||
|---|--------|------------|-------------------|
|
||||
| 1 | `panorama_totales_financieros_principales` | Card principal | 1 fila, 3 campos |
|
||||
| 2 | `panorama_totales_ingreso_compra` | TotalesIngresoCompra | 1 fila, 14 campos |
|
||||
| 3 | `panorama_totales_monetarios` | TotalesMonetarios | 1 fila, 14 campos |
|
||||
| 4 | `panorama_totales_verde` | TotalesVerde | 1 fila, 6 campos |
|
||||
| 5 | `panorama_secos_vendidos` | SecosVendidos | 1 fila, 4 campos |
|
||||
| 6 | `panorama_rechazos_subproductos` | RechazosSubproductos | 6 filas (por tipo) |
|
||||
| 7 | `panorama_serie_temporal_diaria` | Gráficas | Múltiples filas |
|
||||
| 8 | `panorama_top_clientes` | TopClientes | 20 filas max |
|
||||
| 9 | `panorama_conteo_registros` | Footer de filtros | 1 fila, 4 campos |
|
||||
|
||||
**Total: 9 Queries**
|
||||
|
||||
---
|
||||
|
||||
## 🚀 INTEGRACIÓN EN VUE
|
||||
|
||||
```typescript
|
||||
// panorama.vue
|
||||
|
||||
const metrics = ref({
|
||||
financieros: null,
|
||||
ingresoCompra: null,
|
||||
monetarios: null,
|
||||
verde: null,
|
||||
secosVendidos: null,
|
||||
rechazos: [],
|
||||
serieTemporal: [],
|
||||
topClientes: [],
|
||||
conteos: null
|
||||
})
|
||||
|
||||
async function loadAllData() {
|
||||
const params = {
|
||||
fecha_desde: fechaDesde.value,
|
||||
fecha_hasta: fechaHasta.value,
|
||||
incluir_anulados: includeAnulados.value
|
||||
}
|
||||
|
||||
const [
|
||||
financieros,
|
||||
ingresoCompra,
|
||||
monetarios,
|
||||
verde,
|
||||
secosVendidos,
|
||||
rechazos,
|
||||
serieTemporal,
|
||||
topClientes,
|
||||
conteos
|
||||
] = await Promise.all([
|
||||
$fetch('/api/metabase/question/1', { query: params }),
|
||||
$fetch('/api/metabase/question/2', { query: params }),
|
||||
$fetch('/api/metabase/question/3', { query: params }),
|
||||
$fetch('/api/metabase/question/4', { query: params }),
|
||||
$fetch('/api/metabase/question/5', { query: params }),
|
||||
$fetch('/api/metabase/question/6', { query: params }),
|
||||
$fetch('/api/metabase/question/7', { query: params }),
|
||||
$fetch('/api/metabase/question/8', { query: params }),
|
||||
$fetch('/api/metabase/question/9', { query: params })
|
||||
])
|
||||
|
||||
// Asignar directamente, SIN procesamiento
|
||||
metrics.value.financieros = financieros.data.rows[0]
|
||||
metrics.value.ingresoCompra = ingresoCompra.data.rows[0]
|
||||
metrics.value.monetarios = monetarios.data.rows[0]
|
||||
metrics.value.verde = verde.data.rows[0]
|
||||
metrics.value.secosVendidos = secosVendidos.data.rows[0]
|
||||
metrics.value.rechazos = rechazos.data.rows
|
||||
metrics.value.serieTemporal = serieTemporal.data.rows
|
||||
metrics.value.topClientes = topClientes.data.rows
|
||||
metrics.value.conteos = conteos.data.rows[0]
|
||||
}
|
||||
```
|
||||
|
||||
```vue
|
||||
<!-- En el componente TotalesMonetarios.vue -->
|
||||
<template>
|
||||
<UCard>
|
||||
<MetricCard
|
||||
label="Inversión en Uva"
|
||||
:value="formatCurrency(metrics.inversion_uva)"
|
||||
/>
|
||||
<!-- etc... -->
|
||||
</UCard>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
defineProps(['metrics']) // Recibe el objeto directamente de la query
|
||||
</script>
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## ⚠️ NOTAS IMPORTANTES
|
||||
|
||||
1. **Sin composables de métricas**: Los composables `useIngresosMetrics` y `useRechazosMetrics` se eliminan completamente.
|
||||
|
||||
2. **Metabase hace TODO**: Cálculos, agregaciones, promedios ponderados, todo en SQL.
|
||||
|
||||
3. **Vue solo renderiza**: Los componentes reciben props con los valores ya calculados.
|
||||
|
||||
4. **Estructura de respuesta de Metabase**:
|
||||
```javascript
|
||||
{
|
||||
data: {
|
||||
rows: [...], // Array de filas
|
||||
cols: [...], // Metadata de columnas
|
||||
rows_truncated: 1000
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
5. **Optimización**: Las 9 queries se ejecutan en **paralelo** con `Promise.all()`.
|
||||
|
||||
6. **Filtros reactivos**: Cuando cambian los filtros, se vuelven a ejecutar las queries.
|
||||
|
||||
---
|
||||
|
||||
**Documento creado:** 2025-10-14
|
||||
**Autor:** Claude Code
|
||||
**Proyecto:** Analítica Núcleo - Panorama Facturador
|
||||
27
nuxt4-app/app/components/MetricBox.vue
Normal file
27
nuxt4-app/app/components/MetricBox.vue
Normal file
@@ -0,0 +1,27 @@
|
||||
<template>
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-3">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">{{ label }}</div>
|
||||
<div class="text-lg font-bold" :class="valueColor">
|
||||
{{ value }}
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const props = defineProps<{
|
||||
label: string
|
||||
value: string
|
||||
color?: 'default' | 'green' | 'yellow' | 'red' | 'blue'
|
||||
}>()
|
||||
|
||||
const valueColor = computed(() => {
|
||||
const colors = {
|
||||
default: 'text-[var(--brand-text)]',
|
||||
green: 'text-green-400',
|
||||
yellow: 'text-yellow-400',
|
||||
red: 'text-red-400',
|
||||
blue: 'text-blue-400'
|
||||
}
|
||||
return colors[props.color || 'default']
|
||||
})
|
||||
</script>
|
||||
74
nuxt4-app/app/components/SecosVendidos.vue
Normal file
74
nuxt4-app/app/components/SecosVendidos.vue
Normal file
@@ -0,0 +1,74 @@
|
||||
<template>
|
||||
<UCard class="brand-card border border-transparent">
|
||||
<template #header>
|
||||
<h2 class="text-xl font-bold brand-section-title">Café Seco - Inventario y Proyecciones</h2>
|
||||
</template>
|
||||
|
||||
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-3">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">QQ Seco por Vender</div>
|
||||
<div class="text-2xl font-bold text-[var(--brand-text)]">
|
||||
{{ formatNumber(data.total_qq_seco_por_vender) }} QQ
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-3">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">Precio Venta Promedio/QQ</div>
|
||||
<div class="text-2xl font-bold text-green-400">
|
||||
{{ formatCurrency(data.precio_venta_promedio_por_qq) }}
|
||||
</div>
|
||||
<div v-if="data.precio_venta_promedio_por_qq === 0" class="text-xs text-[var(--brand-text-muted)] mt-1">
|
||||
Sin ventas registradas
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-3">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">Precio Compra Promedio/QQ</div>
|
||||
<div class="text-2xl font-bold text-[var(--brand-text)]">
|
||||
{{ formatCurrency(data.precio_compra_promedio_por_qq) }}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-3">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">Margen de Ganancia/QQ</div>
|
||||
<div class="text-2xl font-bold" :class="margenColor">
|
||||
{{ formatCurrency(data.margen_ganancia_por_qq) }}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</UCard>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const props = defineProps<{
|
||||
data: {
|
||||
total_qq_seco_por_vender: number
|
||||
precio_venta_promedio_por_qq: number
|
||||
precio_compra_promedio_por_qq: number
|
||||
margen_ganancia_por_qq: number
|
||||
}
|
||||
}>()
|
||||
|
||||
const margenColor = computed(() => {
|
||||
if (props.data.margen_ganancia_por_qq > 0) return 'text-green-400'
|
||||
if (props.data.margen_ganancia_por_qq < 0) return 'text-red-400'
|
||||
return 'text-[var(--brand-text)]'
|
||||
})
|
||||
|
||||
const formatNumber = (value: number) => {
|
||||
return new Intl.NumberFormat('es-HN', {
|
||||
minimumFractionDigits: 2,
|
||||
maximumFractionDigits: 2
|
||||
}).format(value)
|
||||
}
|
||||
|
||||
const formatCurrency = (value: number) => {
|
||||
if (!value) return 'L 0.00'
|
||||
return new Intl.NumberFormat('es-HN', {
|
||||
style: 'currency',
|
||||
currency: 'HNL',
|
||||
minimumFractionDigits: 2,
|
||||
maximumFractionDigits: 2
|
||||
}).format(value).replace('HNL', 'L')
|
||||
}
|
||||
</script>
|
||||
78
nuxt4-app/app/components/TotalesIngresoCompra.vue
Normal file
78
nuxt4-app/app/components/TotalesIngresoCompra.vue
Normal file
@@ -0,0 +1,78 @@
|
||||
<template>
|
||||
<UCard class="brand-card border border-transparent">
|
||||
<template #header>
|
||||
<h2 class="text-xl font-bold brand-section-title">Totales de Ingreso y Compra</h2>
|
||||
</template>
|
||||
|
||||
<div class="space-y-6">
|
||||
<!-- Uva -->
|
||||
<div>
|
||||
<h3 class="text-lg font-semibold text-[var(--brand-primary)] mb-3">Uva</h3>
|
||||
<div class="grid grid-cols-1 md:grid-cols-4 gap-3">
|
||||
<MetricBox label="LB Uva Ingresada" :value="formatNumber(data.total_lb_uva_ingresada) + ' lb'" />
|
||||
<MetricBox label="QQ Seco Ingresado" :value="formatNumber(data.total_qq_seco_uva_ingresado) + ' QQ'" />
|
||||
<MetricBox label="LB Uva Pagada" :value="formatNumber(data.total_lb_uva_pagada) + ' lb'" color="green" />
|
||||
<MetricBox label="LB Uva en Depósito" :value="formatNumber(data.total_lb_uva_deposito) + ' lb'" color="yellow" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Mojado -->
|
||||
<div>
|
||||
<h3 class="text-lg font-semibold text-[var(--brand-primary)] mb-3">Mojado</h3>
|
||||
<div class="grid grid-cols-1 md:grid-cols-3 gap-3">
|
||||
<MetricBox label="QQ Seco Ingresado" :value="formatNumber(data.total_qq_seco_mojado_ingresado) + ' QQ'" />
|
||||
<MetricBox label="QQ Seco Pagado" :value="formatNumber(data.total_qq_seco_mojado_pagado) + ' QQ'" color="green" />
|
||||
<MetricBox label="QQ en Depósito" :value="formatNumber(data.total_qq_mojado_deposito) + ' QQ'" color="yellow" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Oreado -->
|
||||
<div>
|
||||
<h3 class="text-lg font-semibold text-[var(--brand-primary)] mb-3">Oreado</h3>
|
||||
<div class="grid grid-cols-1 md:grid-cols-3 gap-3">
|
||||
<MetricBox label="QQ Seco Ingresado" :value="formatNumber(data.total_qq_seco_oreado_ingresado) + ' QQ'" />
|
||||
<MetricBox label="QQ Seco Pagado" :value="formatNumber(data.total_qq_seco_oreado_pagado) + ' QQ'" color="green" />
|
||||
<MetricBox label="QQ en Depósito" :value="formatNumber(data.total_qq_oreado_deposito) + ' QQ'" color="yellow" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Totales Generales -->
|
||||
<div class="pt-4 border-t border-[#3a2a16]">
|
||||
<h3 class="text-lg font-semibold text-[var(--brand-primary)] mb-3">Totales Generales</h3>
|
||||
<div class="grid grid-cols-1 md:grid-cols-3 gap-3">
|
||||
<MetricBox label="QQ Seco Ingresado" :value="formatNumber(data.total_qq_seco_ingresado) + ' QQ'" />
|
||||
<MetricBox label="QQ Seco Comprado" :value="formatNumber(data.total_qq_seco_comprado) + ' QQ'" color="green" />
|
||||
<MetricBox label="QQ Seco en Depósito" :value="formatNumber(data.total_qq_seco_deposito) + ' QQ'" color="yellow" />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</UCard>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const props = defineProps<{
|
||||
data: {
|
||||
total_lb_uva_ingresada: number
|
||||
total_qq_seco_uva_ingresado: number
|
||||
total_qq_seco_mojado_ingresado: number
|
||||
total_qq_seco_oreado_ingresado: number
|
||||
total_qq_seco_ingresado: number
|
||||
total_lb_uva_pagada: number
|
||||
total_qq_seco_uva_pagado: number
|
||||
total_qq_seco_mojado_pagado: number
|
||||
total_qq_seco_oreado_pagado: number
|
||||
total_qq_seco_comprado: number
|
||||
total_lb_uva_deposito: number
|
||||
total_qq_mojado_deposito: number
|
||||
total_qq_oreado_deposito: number
|
||||
total_qq_seco_deposito: number
|
||||
}
|
||||
}>()
|
||||
|
||||
const formatNumber = (value: number) => {
|
||||
return new Intl.NumberFormat('es-HN', {
|
||||
minimumFractionDigits: 2,
|
||||
maximumFractionDigits: 2
|
||||
}).format(value)
|
||||
}
|
||||
</script>
|
||||
73
nuxt4-app/app/components/TotalesMonetarios.vue
Normal file
73
nuxt4-app/app/components/TotalesMonetarios.vue
Normal file
@@ -0,0 +1,73 @@
|
||||
<template>
|
||||
<UCard class="brand-card border border-transparent">
|
||||
<template #header>
|
||||
<h2 class="text-xl font-bold brand-section-title">Totales Monetarios</h2>
|
||||
</template>
|
||||
|
||||
<div class="space-y-6">
|
||||
<!-- Inversión Hasta la Fecha -->
|
||||
<div>
|
||||
<h3 class="text-lg font-semibold text-[var(--brand-primary)] mb-3">Inversión Hasta la Fecha</h3>
|
||||
<div class="grid grid-cols-1 md:grid-cols-4 gap-3">
|
||||
<MetricBox label="Inversión Uva" :value="formatCurrency(data.inversion_uva)" />
|
||||
<MetricBox label="Inversión Mojado" :value="formatCurrency(data.inversion_mojado)" />
|
||||
<MetricBox label="Inversión Oreado" :value="formatCurrency(data.inversion_oreado)" />
|
||||
<MetricBox label="Total Invertido" :value="formatCurrency(data.total_invertido)" color="green" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Precios Promedio Ponderados -->
|
||||
<div>
|
||||
<h3 class="text-lg font-semibold text-[var(--brand-primary)] mb-3">Precios Promedio Ponderados</h3>
|
||||
<div class="grid grid-cols-1 md:grid-cols-5 gap-3">
|
||||
<MetricBox label="Uva por LB" :value="formatCurrency(data.precio_promedio_uva_por_lb)" />
|
||||
<MetricBox label="Uva por QQ" :value="formatCurrency(data.precio_promedio_uva_por_qq)" />
|
||||
<MetricBox label="Mojado por QQ" :value="formatCurrency(data.precio_promedio_mojado_por_qq)" />
|
||||
<MetricBox label="Oreado por QQ" :value="formatCurrency(data.precio_promedio_oreado_por_qq)" />
|
||||
<MetricBox label="Global QQ Seco" :value="formatCurrency(data.precio_promedio_qq_seco)" color="blue" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Inversión Restante Esperada -->
|
||||
<div>
|
||||
<h3 class="text-lg font-semibold text-[var(--brand-primary)] mb-3">Inversión Restante Esperada</h3>
|
||||
<div class="grid grid-cols-1 md:grid-cols-4 gap-3">
|
||||
<MetricBox label="Restante Uva" :value="formatCurrency(data.inversion_restante_uva)" color="yellow" />
|
||||
<MetricBox label="Restante Mojado" :value="formatCurrency(data.inversion_restante_mojado)" color="yellow" />
|
||||
<MetricBox label="Restante Oreado" :value="formatCurrency(data.inversion_restante_oreado)" color="yellow" />
|
||||
<MetricBox label="Total Restante" :value="formatCurrency(data.inversion_restante_esperada)" color="red" />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</UCard>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const props = defineProps<{
|
||||
data: {
|
||||
inversion_uva: number
|
||||
inversion_mojado: number
|
||||
inversion_oreado: number
|
||||
total_invertido: number
|
||||
precio_promedio_uva_por_lb: number
|
||||
precio_promedio_uva_por_qq: number
|
||||
precio_promedio_mojado_por_qq: number
|
||||
precio_promedio_oreado_por_qq: number
|
||||
precio_promedio_qq_seco: number
|
||||
inversion_restante_uva: number
|
||||
inversion_restante_mojado: number
|
||||
inversion_restante_oreado: number
|
||||
inversion_restante_esperada: number
|
||||
}
|
||||
}>()
|
||||
|
||||
const formatCurrency = (value: number) => {
|
||||
if (!value) return 'L 0.00'
|
||||
return new Intl.NumberFormat('es-HN', {
|
||||
style: 'currency',
|
||||
currency: 'HNL',
|
||||
minimumFractionDigits: 2,
|
||||
maximumFractionDigits: 2
|
||||
}).format(value).replace('HNL', 'L')
|
||||
}
|
||||
</script>
|
||||
81
nuxt4-app/app/components/TotalesVerde.vue
Normal file
81
nuxt4-app/app/components/TotalesVerde.vue
Normal file
@@ -0,0 +1,81 @@
|
||||
<template>
|
||||
<UCard class="brand-card border border-transparent">
|
||||
<template #header>
|
||||
<h2 class="text-xl font-bold brand-section-title">Café Verde</h2>
|
||||
</template>
|
||||
|
||||
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-3">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">LB Neto Verde Total</div>
|
||||
<div class="text-2xl font-bold text-[var(--brand-text)]">
|
||||
{{ formatNumber(data.total_lb_neto_verde) }} lb
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-3">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">LB Neto Comprado</div>
|
||||
<div class="text-2xl font-bold text-green-400">
|
||||
{{ formatNumber(data.total_lb_neto_comprado_verde) }} lb
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-3">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">LB Neto en Depósito</div>
|
||||
<div class="text-2xl font-bold text-yellow-400">
|
||||
{{ formatNumber(data.total_lb_neto_verde_deposito) }} lb
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-3">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">Precio Promedio Pagado</div>
|
||||
<div class="text-2xl font-bold text-[var(--brand-text)]">
|
||||
{{ formatCurrency(data.precio_promedio_verde_pagado) }}/lb
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-3">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">Inversión Hasta la Fecha</div>
|
||||
<div class="text-2xl font-bold text-green-400">
|
||||
{{ formatCurrency(data.inversion_verde_hasta_fecha) }}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-3">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">Inversión Restante</div>
|
||||
<div class="text-2xl font-bold text-yellow-400">
|
||||
{{ formatCurrency(data.inversion_restante_verde) }}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</UCard>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const props = defineProps<{
|
||||
data: {
|
||||
total_lb_neto_verde: number
|
||||
precio_promedio_verde_pagado: number
|
||||
total_lb_neto_verde_deposito: number
|
||||
inversion_verde_hasta_fecha: number
|
||||
inversion_restante_verde: number
|
||||
total_lb_neto_comprado_verde: number
|
||||
}
|
||||
}>()
|
||||
|
||||
const formatNumber = (value: number) => {
|
||||
return new Intl.NumberFormat('es-HN', {
|
||||
minimumFractionDigits: 2,
|
||||
maximumFractionDigits: 2
|
||||
}).format(value)
|
||||
}
|
||||
|
||||
const formatCurrency = (value: number) => {
|
||||
if (!value) return 'L 0.00'
|
||||
return new Intl.NumberFormat('es-HN', {
|
||||
style: 'currency',
|
||||
currency: 'HNL',
|
||||
minimumFractionDigits: 2,
|
||||
maximumFractionDigits: 2
|
||||
}).format(value).replace('HNL', 'L')
|
||||
}
|
||||
</script>
|
||||
82
nuxt4-app/app/components/rechazos/RechazosRechazoCard.vue
Normal file
82
nuxt4-app/app/components/rechazos/RechazosRechazoCard.vue
Normal file
@@ -0,0 +1,82 @@
|
||||
<template>
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] p-4">
|
||||
<div class="flex items-center justify-between mb-3">
|
||||
<h3 class="text-lg font-semibold text-[var(--brand-text)]">{{ title }}</h3>
|
||||
<div :class="colorClasses" class="w-3 h-3 rounded-full"></div>
|
||||
</div>
|
||||
|
||||
<div class="space-y-2">
|
||||
<div class="flex justify-between items-center">
|
||||
<span class="text-xs text-[var(--brand-text-muted)]">Cantidad</span>
|
||||
<span class="text-sm font-medium text-[var(--brand-text)]">
|
||||
{{ formatNumber(data.total_cantidad) }} {{ unidad }}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div class="flex justify-between items-center">
|
||||
<span class="text-xs text-[var(--brand-text-muted)]">Total Cobrado</span>
|
||||
<span class="text-sm font-bold text-green-400">
|
||||
{{ formatCurrency(data.total_cobrado) }}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div class="flex justify-between items-center">
|
||||
<span class="text-xs text-[var(--brand-text-muted)]">Precio Promedio</span>
|
||||
<span class="text-sm font-medium text-[var(--brand-text)]">
|
||||
{{ formatCurrency(data.precio_promedio) }}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div class="flex justify-between items-center pt-2 border-t border-[#3a2a16]">
|
||||
<span class="text-xs text-[var(--brand-text-muted)]">Registros</span>
|
||||
<span class="text-xs font-medium text-[var(--brand-text)]">
|
||||
{{ data.num_registros }}
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const props = defineProps<{
|
||||
title: string
|
||||
data: {
|
||||
tipo: string
|
||||
num_registros: number
|
||||
total_cantidad: number
|
||||
total_cobrado: number
|
||||
precio_promedio: number
|
||||
}
|
||||
unidad: string
|
||||
color: string
|
||||
}>()
|
||||
|
||||
const colorClasses = computed(() => {
|
||||
const colors: Record<string, string> = {
|
||||
blue: 'bg-blue-500',
|
||||
green: 'bg-green-500',
|
||||
yellow: 'bg-yellow-500',
|
||||
red: 'bg-red-500',
|
||||
purple: 'bg-purple-500',
|
||||
pink: 'bg-pink-500'
|
||||
}
|
||||
return colors[props.color] || 'bg-gray-500'
|
||||
})
|
||||
|
||||
const formatNumber = (value: number) => {
|
||||
return new Intl.NumberFormat('es-HN', {
|
||||
minimumFractionDigits: 2,
|
||||
maximumFractionDigits: 2
|
||||
}).format(value)
|
||||
}
|
||||
|
||||
const formatCurrency = (value: number) => {
|
||||
if (!value) return 'L 0.00'
|
||||
return new Intl.NumberFormat('es-HN', {
|
||||
style: 'currency',
|
||||
currency: 'HNL',
|
||||
minimumFractionDigits: 2,
|
||||
maximumFractionDigits: 2
|
||||
}).format(value).replace('HNL', 'L')
|
||||
}
|
||||
</script>
|
||||
@@ -5,62 +5,71 @@
|
||||
<h2 class="text-xl font-bold brand-section-title">Rechazos y Subproductos</h2>
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-4 py-2">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">Total Rechazos</div>
|
||||
<div class="text-2xl font-bold text-[var(--brand-text)]">{{ formatCurrency(totalRechazos.value) }}</div>
|
||||
<div class="text-2xl font-bold text-[var(--brand-text)]">{{ formatCurrency(totalRechazos) }}</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
|
||||
<RechazosRechazoCard
|
||||
title="Chibolita"
|
||||
:metrics="metrics.chibolita"
|
||||
unidad="libras"
|
||||
color="blue"
|
||||
/>
|
||||
<RechazosRechazoCard
|
||||
title="Perico"
|
||||
:metrics="metrics.perico"
|
||||
unidad="libras"
|
||||
color="green"
|
||||
/>
|
||||
<RechazosRechazoCard
|
||||
title="Vano"
|
||||
:metrics="metrics.vano"
|
||||
unidad="galones"
|
||||
color="yellow"
|
||||
/>
|
||||
<RechazosRechazoCard
|
||||
title="Picadillo"
|
||||
:metrics="metrics.picadillo"
|
||||
unidad="galones"
|
||||
color="red"
|
||||
/>
|
||||
<RechazosRechazoCard
|
||||
title="Magalla"
|
||||
:metrics="metrics.magalla"
|
||||
unidad="galones"
|
||||
color="purple"
|
||||
/>
|
||||
<RechazosRechazoCard
|
||||
title="Pinta"
|
||||
:metrics="metrics.pinta"
|
||||
unidad="libras"
|
||||
color="pink"
|
||||
v-for="rechazo in rechazosFormateados"
|
||||
:key="rechazo.tipo"
|
||||
:title="rechazo.title"
|
||||
:data="rechazo.data"
|
||||
:unidad="rechazo.unidad"
|
||||
:color="rechazo.color"
|
||||
/>
|
||||
</div>
|
||||
</UCard>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import type { RechazosMetrics } from '~/composables/useRechazosMetrics'
|
||||
|
||||
const props = defineProps<{
|
||||
metrics: RechazosMetrics
|
||||
data: Array<{
|
||||
tipo: string
|
||||
num_registros: number
|
||||
total_cantidad: number
|
||||
total_cobrado: number
|
||||
precio_promedio: number
|
||||
}>
|
||||
}>()
|
||||
|
||||
const totalRechazos = computed(() => props.metrics.totalRechazos)
|
||||
// Calcular total de rechazos
|
||||
const totalRechazos = computed(() => {
|
||||
return props.data.reduce((sum, r) => sum + (r.total_cobrado || 0), 0)
|
||||
})
|
||||
|
||||
// Formatear rechazos para los cards
|
||||
const rechazosFormateados = computed(() => {
|
||||
const config = {
|
||||
chibolita: { title: 'Chibolita', unidad: 'libras', color: 'blue' },
|
||||
perico: { title: 'Perico', unidad: 'libras', color: 'green' },
|
||||
vano: { title: 'Vano', unidad: 'galones', color: 'yellow' },
|
||||
picadillo: { title: 'Picadillo', unidad: 'galones', color: 'red' },
|
||||
magalla: { title: 'Magalla', unidad: 'galones', color: 'purple' },
|
||||
pinta: { title: 'Pinta', unidad: 'libras', color: 'pink' }
|
||||
}
|
||||
|
||||
return Object.entries(config).map(([tipo, cfg]) => {
|
||||
const rechazo = props.data.find(r => r.tipo === tipo)
|
||||
return {
|
||||
tipo,
|
||||
title: cfg.title,
|
||||
unidad: cfg.unidad,
|
||||
color: cfg.color,
|
||||
data: rechazo || {
|
||||
tipo,
|
||||
num_registros: 0,
|
||||
total_cantidad: 0,
|
||||
total_cobrado: 0,
|
||||
precio_promedio: 0
|
||||
}
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
const formatCurrency = (value: number) => {
|
||||
if (!value) return 'L 0.00'
|
||||
return new Intl.NumberFormat('es-HN', {
|
||||
style: 'currency',
|
||||
currency: 'HNL',
|
||||
@@ -68,4 +77,4 @@ const formatCurrency = (value: number) => {
|
||||
maximumFractionDigits: 2
|
||||
}).format(value).replace('HNL', 'L')
|
||||
}
|
||||
</script>
|
||||
</script>
|
||||
|
||||
@@ -1,15 +1,248 @@
|
||||
<template>
|
||||
<MaintenanceMode
|
||||
title="Panorama Facturador"
|
||||
description="El panel de panorama facturador está temporalmente deshabilitado mientras actualizamos las métricas."
|
||||
icon="i-lucide-bar-chart-3"
|
||||
technical-info="Dashboard de facturación en proceso de integración con nueva fuente de datos empresariales."
|
||||
/>
|
||||
<div class="flex flex-col gap-8">
|
||||
<!-- Loading State -->
|
||||
<UCard v-if="loading && !data" class="brand-card border border-transparent">
|
||||
<div class="flex flex-col items-center justify-center gap-4 py-10 text-[var(--brand-text-muted)]">
|
||||
<div class="flex items-center gap-3">
|
||||
<span class="inline-flex h-8 w-8 animate-spin rounded-full border-2 border-[#c08040] border-t-transparent align-middle" aria-hidden="true" />
|
||||
<span class="text-sm uppercase tracking-[0.3em]">Cargando datos...</span>
|
||||
</div>
|
||||
</div>
|
||||
</UCard>
|
||||
|
||||
<!-- Error State -->
|
||||
<div v-else-if="error" class="rounded-lg border border-red-500/40 bg-red-500/18 p-4 text-sm text-red-200">
|
||||
<p>Error al cargar datos: {{ error }}</p>
|
||||
<UButton class="mt-4" @click="loadData" color="primary">
|
||||
Reintentar
|
||||
</UButton>
|
||||
</div>
|
||||
|
||||
<!-- Main Content -->
|
||||
<template v-else-if="data">
|
||||
<!-- Card de Filtros -->
|
||||
<UCard class="brand-card border border-transparent">
|
||||
<template #header>
|
||||
<div class="flex flex-col gap-3">
|
||||
<div class="flex flex-wrap items-center justify-between gap-3">
|
||||
<div>
|
||||
<h2 class="text-xl font-bold brand-section-title">Filtros y Configuraciones</h2>
|
||||
<p class="text-xs text-[var(--brand-text-muted)] mt-1">
|
||||
Aplicados a <code>created_at</code> de ingresos y rechazos
|
||||
</p>
|
||||
</div>
|
||||
<div class="flex items-center gap-2">
|
||||
<UCheckbox v-model="includeAnulados" label="Incluir anulados" @update:model-value="onToggleAnulados" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Alerta roja cuando incluye anulados -->
|
||||
<UAlert
|
||||
v-if="includeAnulados"
|
||||
color="error"
|
||||
variant="solid"
|
||||
icon="i-lucide-alert-triangle"
|
||||
title="Incluir anulados activado"
|
||||
description="Los cálculos incluyen registros anulados. Esto puede afectar los resultados financieros."
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<DateRangeSelector
|
||||
:selected-preset="selectedPreset"
|
||||
:fecha-desde="fechaDesde"
|
||||
:fecha-hasta="fechaHasta"
|
||||
@update:selected-preset="onUpdatePreset"
|
||||
@update:fecha-desde="onUpdateFechaDesde"
|
||||
@update:fecha-hasta="onUpdateFechaHasta"
|
||||
/>
|
||||
|
||||
<template #footer>
|
||||
<div class="text-xs text-[var(--brand-text-muted)]">
|
||||
Rango activo: {{ rangoLegible }} · Registros considerados: Ingresos {{ data.conteos.ingresos_filtrados }}/{{ data.conteos.ingresos_total }} · Rechazos {{ data.conteos.rechazos_filtrados }}/{{ data.conteos.rechazos_total }}
|
||||
</div>
|
||||
</template>
|
||||
</UCard>
|
||||
|
||||
<!-- Totales Financieros - Resumen Principal -->
|
||||
<UCard class="brand-card border border-transparent">
|
||||
<template #header>
|
||||
<div class="flex items-center justify-between">
|
||||
<div>
|
||||
<h2 class="text-2xl font-bold brand-section-title">Totales Financieros</h2>
|
||||
<p class="text-sm text-[var(--brand-text-muted)] mt-1">Vista general de ingresos, inversiones y rechazos</p>
|
||||
</div>
|
||||
<UButton
|
||||
:loading="loading"
|
||||
:ui="{ base: 'bg-[#c08040] text-[#1b1209] border border-[#d99a56] hover:bg-[#d99a56] hover:border-[#f0c07c]' }"
|
||||
size="sm"
|
||||
@click="loadData"
|
||||
>
|
||||
<template #leading>
|
||||
<UIcon name="i-lucide-refresh-cw" :class="{ 'animate-spin': loading }" />
|
||||
</template>
|
||||
Actualizar
|
||||
</UButton>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<div class="grid grid-cols-1 md:grid-cols-3 gap-4">
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-6 py-4">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">Total Invertido en Café</div>
|
||||
<div class="text-3xl font-bold text-[var(--brand-primary)]">
|
||||
{{ formatCurrency(data.financieros.total_invertido_cafe) }}
|
||||
</div>
|
||||
</div>
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-6 py-4">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">Total Rechazos</div>
|
||||
<div class="text-3xl font-bold text-green-400">
|
||||
{{ formatCurrency(data.financieros.total_rechazos) }}
|
||||
</div>
|
||||
</div>
|
||||
<div class="rounded-lg border border-[#3a2a16] bg-[#1c140c] px-6 py-4">
|
||||
<div class="text-xs text-[var(--brand-text-muted)] uppercase tracking-wide mb-1">Balance Neto</div>
|
||||
<div class="text-3xl font-bold text-[var(--brand-text)]">
|
||||
{{ formatCurrency(data.financieros.balance_neto) }}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<template #footer>
|
||||
<div class="text-xs text-[var(--brand-text-muted)]">
|
||||
Última actualización: {{ lastUpdated }}
|
||||
</div>
|
||||
</template>
|
||||
</UCard>
|
||||
|
||||
<!-- Secciones de Ingresos -->
|
||||
<SecosVendidos :data="data.secosVendidos" />
|
||||
<TotalesIngresoCompra :data="data.ingresoCompra" />
|
||||
<TotalesMonetarios :data="data.monetarios" />
|
||||
<TotalesVerde :data="data.verde" />
|
||||
|
||||
<!-- Sección de Rechazos -->
|
||||
<RechazosSubproductos :data="data.rechazos" />
|
||||
</template>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
// Define page metadata
|
||||
definePageMeta({
|
||||
layout: 'informe',
|
||||
title: 'Panorama Facturador'
|
||||
})
|
||||
|
||||
// Reactive state
|
||||
const data = ref<any>(null)
|
||||
const loading = ref(false)
|
||||
const error = ref<string | null>(null)
|
||||
const lastUpdated = ref<string>('')
|
||||
|
||||
// Filtros
|
||||
const includeAnulados = ref(false)
|
||||
|
||||
type PresetValue =
|
||||
| '' | 'custom' | 'hoy' | 'semana' | 'mes' | 'ytd'
|
||||
| 'cosecha-20-21' | 'cosecha-21-22' | 'cosecha-22-23'
|
||||
| 'cosecha-23-24' | 'cosecha-24-25' | 'cosecha-25-26'
|
||||
|
||||
const selectedPreset = ref<PresetValue>('cosecha-25-26')
|
||||
const fechaDesde = ref<string | null>(null)
|
||||
const fechaHasta = ref<string | null>(null)
|
||||
|
||||
// Computed
|
||||
const rangoLegible = computed(() => {
|
||||
if (!fechaDesde.value && !fechaHasta.value) return 'Sin filtro de fecha'
|
||||
const f = fechaDesde.value ?? '—'
|
||||
const t = fechaHasta.value ?? '—'
|
||||
return `${f} → ${t}`
|
||||
})
|
||||
|
||||
// Format currency helper
|
||||
const formatCurrency = (value: number) => {
|
||||
if (!value) return 'L 0.00'
|
||||
return new Intl.NumberFormat('es-HN', {
|
||||
style: 'currency',
|
||||
currency: 'HNL',
|
||||
minimumFractionDigits: 2,
|
||||
maximumFractionDigits: 2
|
||||
}).format(value).replace('HNL', 'L')
|
||||
}
|
||||
|
||||
// Methods
|
||||
async function loadData() {
|
||||
loading.value = true
|
||||
error.value = null
|
||||
|
||||
try {
|
||||
const result = await $fetch('/api/metabase/panorama', {
|
||||
method: 'POST',
|
||||
body: {
|
||||
fecha_desde: fechaDesde.value,
|
||||
fecha_hasta: fechaHasta.value,
|
||||
incluir_anulados: includeAnulados.value
|
||||
}
|
||||
})
|
||||
|
||||
data.value = result
|
||||
lastUpdated.value = new Date().toLocaleString('es-ES', {
|
||||
year: 'numeric',
|
||||
month: 'long',
|
||||
day: 'numeric',
|
||||
hour: '2-digit',
|
||||
minute: '2-digit'
|
||||
})
|
||||
} catch (err: any) {
|
||||
error.value = err.message || 'Error al cargar datos'
|
||||
console.error('Error loading panorama data:', err)
|
||||
} finally {
|
||||
loading.value = false
|
||||
}
|
||||
}
|
||||
|
||||
async function onToggleAnulados(newValue: boolean | 'indeterminate') {
|
||||
if (newValue === true) {
|
||||
// Pedir confirmación al activar
|
||||
const confirmed = confirm(
|
||||
'⚠️ ADVERTENCIA\n\n' +
|
||||
'Está a punto de incluir registros ANULADOS en los cálculos.\n\n' +
|
||||
'Esto puede afectar significativamente los resultados financieros y métricas.\n\n' +
|
||||
'¿Está seguro de que desea continuar?'
|
||||
)
|
||||
|
||||
if (!confirmed) {
|
||||
// Si cancela, revertir el cambio
|
||||
includeAnulados.value = false
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
// Recargar datos con el nuevo valor
|
||||
await loadData()
|
||||
}
|
||||
|
||||
function onUpdatePreset(value: PresetValue) {
|
||||
selectedPreset.value = value
|
||||
}
|
||||
|
||||
function onUpdateFechaDesde(value: string | null) {
|
||||
fechaDesde.value = value
|
||||
}
|
||||
|
||||
function onUpdateFechaHasta(value: string | null) {
|
||||
fechaHasta.value = value
|
||||
}
|
||||
|
||||
// Watchers - reload data when filters change
|
||||
watch([fechaDesde, fechaHasta], () => {
|
||||
loadData()
|
||||
})
|
||||
|
||||
// Load data on mount
|
||||
onMounted(() => {
|
||||
// Default preset: cosecha 25-26
|
||||
selectedPreset.value = 'cosecha-25-26'
|
||||
loadData()
|
||||
})
|
||||
</script>
|
||||
|
||||
145
nuxt4-app/server/api/metabase/panorama.post.ts
Normal file
145
nuxt4-app/server/api/metabase/panorama.post.ts
Normal file
@@ -0,0 +1,145 @@
|
||||
/**
|
||||
* Execute all panorama queries in parallel
|
||||
* Returns data for the Panorama Facturador page
|
||||
*/
|
||||
export default defineEventHandler(async (event) => {
|
||||
const body = await readBody(event)
|
||||
|
||||
const { fecha_desde = null, fecha_hasta = null, incluir_anulados = false } = body
|
||||
|
||||
try {
|
||||
// First, get all cards to find our panorama queries
|
||||
const allCards = await getMetabaseCards('all')
|
||||
|
||||
// Find our panorama queries by name
|
||||
const queryNames = [
|
||||
'panorama_totales_financieros_principales',
|
||||
'panorama_totales_ingreso_compra',
|
||||
'panorama_totales_monetarios',
|
||||
'panorama_totales_verde',
|
||||
'panorama_secos_vendidos',
|
||||
'panorama_rechazos_subproductos',
|
||||
'panorama_serie_temporal_diaria',
|
||||
'panorama_top_clientes',
|
||||
'panorama_conteo_registros'
|
||||
]
|
||||
|
||||
const cards: Record<string, any> = {}
|
||||
|
||||
for (const name of queryNames) {
|
||||
const card = allCards.find((c: any) => c.name === name)
|
||||
if (!card) {
|
||||
console.warn(`[Panorama] Query not found: ${name}`)
|
||||
} else {
|
||||
cards[name] = card
|
||||
}
|
||||
}
|
||||
|
||||
// Build parameters array for Metabase queries
|
||||
const parameters = [
|
||||
{
|
||||
type: 'date/single',
|
||||
target: ['variable', ['template-tag', 'fecha_desde']],
|
||||
value: fecha_desde
|
||||
},
|
||||
{
|
||||
type: 'date/single',
|
||||
target: ['variable', ['template-tag', 'fecha_hasta']],
|
||||
value: fecha_hasta
|
||||
},
|
||||
{
|
||||
type: 'category',
|
||||
target: ['variable', ['template-tag', 'incluir_anulados']],
|
||||
value: incluir_anulados
|
||||
}
|
||||
]
|
||||
|
||||
// Execute all queries in parallel
|
||||
const [
|
||||
financieros,
|
||||
ingresoCompra,
|
||||
monetarios,
|
||||
verde,
|
||||
secosVendidos,
|
||||
rechazos,
|
||||
serieTemporal,
|
||||
topClientes,
|
||||
conteos
|
||||
] = await Promise.all([
|
||||
cards['panorama_totales_financieros_principales']
|
||||
? executeCardQuery(cards['panorama_totales_financieros_principales'].id, parameters)
|
||||
: { data: { rows: [[0, 0, 0]] } },
|
||||
cards['panorama_totales_ingreso_compra']
|
||||
? executeCardQuery(cards['panorama_totales_ingreso_compra'].id, parameters)
|
||||
: { data: { rows: [[]] } },
|
||||
cards['panorama_totales_monetarios']
|
||||
? executeCardQuery(cards['panorama_totales_monetarios'].id, parameters)
|
||||
: { data: { rows: [[]] } },
|
||||
cards['panorama_totales_verde']
|
||||
? executeCardQuery(cards['panorama_totales_verde'].id, parameters)
|
||||
: { data: { rows: [[]] } },
|
||||
cards['panorama_secos_vendidos']
|
||||
? executeCardQuery(cards['panorama_secos_vendidos'].id, parameters)
|
||||
: { data: { rows: [[]] } },
|
||||
cards['panorama_rechazos_subproductos']
|
||||
? executeCardQuery(cards['panorama_rechazos_subproductos'].id, parameters)
|
||||
: { data: { rows: [] } },
|
||||
cards['panorama_serie_temporal_diaria']
|
||||
? executeCardQuery(cards['panorama_serie_temporal_diaria'].id, parameters)
|
||||
: { data: { rows: [] } },
|
||||
cards['panorama_top_clientes']
|
||||
? executeCardQuery(cards['panorama_top_clientes'].id, parameters)
|
||||
: { data: { rows: [] } },
|
||||
cards['panorama_conteo_registros']
|
||||
? executeCardQuery(cards['panorama_conteo_registros'].id, parameters)
|
||||
: { data: { rows: [[0, 0, 0, 0]] } }
|
||||
])
|
||||
|
||||
// Transform Metabase responses to objects for easier frontend consumption
|
||||
const transformSingleRow = (result: any) => {
|
||||
if (!result.data?.rows?.[0] || !result.data?.cols) return {}
|
||||
|
||||
const row = result.data.rows[0]
|
||||
const cols = result.data.cols
|
||||
const obj: any = {}
|
||||
|
||||
cols.forEach((col: any, index: number) => {
|
||||
obj[col.name] = row[index]
|
||||
})
|
||||
|
||||
return obj
|
||||
}
|
||||
|
||||
const transformMultipleRows = (result: any) => {
|
||||
if (!result.data?.rows || !result.data?.cols) return []
|
||||
|
||||
const cols = result.data.cols
|
||||
return result.data.rows.map((row: any[]) => {
|
||||
const obj: any = {}
|
||||
cols.forEach((col: any, index: number) => {
|
||||
obj[col.name] = row[index]
|
||||
})
|
||||
return obj
|
||||
})
|
||||
}
|
||||
|
||||
// Return all data in a structured format
|
||||
return {
|
||||
financieros: transformSingleRow(financieros),
|
||||
ingresoCompra: transformSingleRow(ingresoCompra),
|
||||
monetarios: transformSingleRow(monetarios),
|
||||
verde: transformSingleRow(verde),
|
||||
secosVendidos: transformSingleRow(secosVendidos),
|
||||
rechazos: transformMultipleRows(rechazos),
|
||||
serieTemporal: transformMultipleRows(serieTemporal),
|
||||
topClientes: transformMultipleRows(topClientes),
|
||||
conteos: transformSingleRow(conteos)
|
||||
}
|
||||
} catch (error: any) {
|
||||
console.error('[API] Failed to execute panorama queries:', error)
|
||||
throw createError({
|
||||
statusCode: error.statusCode || 500,
|
||||
statusMessage: error.statusMessage || 'Failed to execute panorama queries'
|
||||
})
|
||||
}
|
||||
})
|
||||
Reference in New Issue
Block a user