49 lines
1.9 KiB
JavaScript
49 lines
1.9 KiB
JavaScript
/**
|
|
* GET /api/en-service
|
|
* Marchés actifs en service (non clôturés)
|
|
*/
|
|
const express = require('express');
|
|
const router = express.Router();
|
|
const { getMarches } = require('../services/baserow');
|
|
const { isCloture, normalizeMarche, parseNum, selectVal } = require('../services/calc');
|
|
|
|
router.get('/', async (req, res) => {
|
|
try {
|
|
const { region, entrepreneur, projet, nature, statut } = req.query;
|
|
const regionFilter = req.regionFilter;
|
|
|
|
let rows = await getMarches();
|
|
|
|
// Uniquement en service (non clôturés, observation = "En service")
|
|
rows = rows.filter(r => !isCloture(r) && selectVal(r.observation).toLowerCase().includes('en service'));
|
|
|
|
// Filtres
|
|
if (regionFilter) rows = rows.filter(r => r.region === regionFilter);
|
|
else if (region) rows = rows.filter(r => r.region === region);
|
|
if (entrepreneur) rows = rows.filter(r => String(r.entrepreneur || '').toLowerCase().includes(entrepreneur.toLowerCase()));
|
|
if (projet) rows = rows.filter(r => String(r.projet || '').toLowerCase().includes(projet.toLowerCase()));
|
|
if (nature) rows = rows.filter(r => String(r.nature || '').toLowerCase().includes(nature.toLowerCase()));
|
|
if (statut) rows = rows.filter(r => String(r.statut || '').toLowerCase().includes(statut.toLowerCase()));
|
|
|
|
const items = rows.map(normalizeMarche);
|
|
|
|
// Agrégats
|
|
const totalBudget = rows.reduce((s, r) => s + parseNum(r.tot_marche ?? r.totmarche ?? r.montant), 0);
|
|
const tauxList = items.map(r => r.taux_phy_raw).filter(v => v > 0);
|
|
const tauxMoyen = tauxList.length
|
|
? Math.round(tauxList.reduce((a, b) => a + b, 0) / tauxList.length * 10) / 10
|
|
: 0;
|
|
|
|
res.json({
|
|
count: items.length,
|
|
budget_total_raw: totalBudget,
|
|
taux_avancement_moyen: tauxMoyen,
|
|
items,
|
|
});
|
|
} catch (err) {
|
|
res.status(502).json({ error: 'Erreur Baserow', detail: err.message });
|
|
}
|
|
});
|
|
|
|
module.exports = router;
|