Le flux Météo France renvoie parfois des phénomènes explicitement au vert
(colorId 1) : ils étaient traités comme des alertes, d'où un « Que faire
maintenant ? » affiché sur un phénomène vert et un dropdown limité aux
phénomènes absents du flux.
- "Active" = vigilance jaune ou plus (colorId >= 2) ; le vert n'est jamais
une alerte
- Tout au vert → un seul encadré « Aucune vigilance active »
- Dropdown regroupe désormais tous les phénomènes au vert (présents au vert
dans le flux + absents)
- Une vigilance présente → bloc complet « Que faire maintenant ? » sur
chacune des alertes actives (au lieu de la première seulement)
- Tableau Aujourd'hui vs Demain et suite : inchangés
Au passage, fix typage FranceMap : Map par défaut typée pour ne pas perdre
VigilanceAlert (5 erreurs astro check), imports inutilisés retirés.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
Les 1-2 derniers jours du flux climato `latest` ont souvent 0-1 station
publiée (parfois TN sans TX), produisant des agrégats département nuls ou
non représentatifs. Symptômes côté page dept : TX/TN vides ou tx==tn à
J-1/J-2 sur le graphe, et moyenne 3j/7j figée (computeAnomaly lisait
climato.days brut et retombait sur les mêmes vieux jours complets).
- climato.ts : exposer la couverture réelle par champ (txN/tnN, optionnels,
rétro-compatibles avec le cache 24h).
- [code].astro : `_mergedDay` choisit par champ la source fiable —
climato si couverture >= 50% de la médiane station du dept, sinon SYNOP
(garde >= 3 obs/jour pour éviter tx==tn), sinon valeur fine, sinon null.
Le badge anomalie utilise désormais les 7 derniers jours complets
(finissant J-1) au lieu de climato.days brut, recollant graphe et badge.
Vérifié par simulation sur données réelles (depts 31/75/13) : jours pleins
inchangés (zéro régression), jours récents partiels comblés proprement.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
Fetcher 48h d'obs horaires (au lieu de 24h) pour avoir le jour J-1 complet.
Dériver TX/TN journaliers (max/min des températures horaires) et les injecter
dans les fenêtres 7j/30j/1an pour les jours absents de la climato CSV.
Le tab "24h" du graphique reste filtré sur les 24 dernières heures.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
TX revient en moyenne inter-stations (comme TN/TM) ; la version MAX était
une mauvaise piste. Les vues 7j/30j/1an sont désormais construites sur une
plage calendaire fixe se terminant à aujourd'hui (heure Paris) : les jours
sans données source (lag publication ~1-4j) apparaissent comme points nuls
plutôt que de décaler la fenêtre vers le passé.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
- Ajoute `hasJ1Period()` — source de vérité réelle au lieu de
s'appuyer sur l'heure de publication (fixe 6h/16h non fiable,
bulletins constatés à 23h et 00h01).
- Home : `tomorrowAvailable` conditionné à `hasJ1Period(snapshot)`
en plus de `currentEcheance === 'J'`.
- Dept : `j1Available` pilote la colonne Demain du tableau de
comparaison et le calcul `tomorrowColor`/`changed`.
- Tooltip J+1 désactivé : suppression de la mention "vers 6h".
- cache.ts / vigilance.ts : `bulletinIsSuperseded` + refresh sync
via `shouldRevalidateSync` (pré-existant, inclus dans le push).
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
Le navigateur cachait l ancienne version HTML → utilisateurs voyaient le
bulletin Vigilance précédent (carte du 25 mai 6h) malgré le bon HTML
servi par le serveur (carte du 25 mai 16h). max-age=60 + must-revalidate
force une revalidation conditionnelle à chaque visite > 1 min.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
- /departement/[code] : Promise.allSettled sur les 3 fetches externes
(vigilance MF, climato data.gouv, hourly SYNOP). Avant : ~15-20s
sériel cold-fetch. Après : ~10s max (= temps du plus lent = climato).
- normales (3 ranges) : Promise.all aussi, économise ~30 ms.
- /api/health enrichi avec vigilance.productDatetime + ageSeconds
pour permettre au cron freshness de checker sans /api/vigilance
(qui a été supprimé en public).
Pré-requis pour le cron warmup côté infra repo (cf. scripts/cron-warmup-info-canicule.sh).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Le bulletin MF est publié 2× par jour (~6h et ~16h Paris). Pendant la nuit
(de minuit au prochain bulletin ~6h), on est techniquement dans le J+1 du
bulletin courant, qui correspond pour l'utilisateur à "aujourd'hui réel".
Avant : on affichait toujours J → la nuit, la carte montrait les alertes
"d'hier" alors que l'utilisateur cherche "celles d'aujourd'hui réel".
Symptôme : Florian voit du orange sur vigilance.meteofrance.fr (qui prend
la bonne écheance) mais pas chez nous (qui restait collé sur J).
Fix :
- currentEcheance(snapshot) compare now au end_time du J : si dépassé, J1.
- index.astro + departement.astro utilisent cette écheance pour
"aujourd'hui". Page dept : "demain" = J1 si on est sur J, vide sinon.
- Labels rendus dynamiques : "Niveau par département — mardi 26 mai" au
lieu de "(aujourd'hui)" générique.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
1. lib/vigilance.ts : provider Météo France officielle via DPVigilance/v1
/cartevigilance/encours. Map vers VigilanceAlert (1 par dept × phen × J/J1)
en prenant phenomenon_max_color_id (pire de la journée). Filtre les domains
non-dept (FRA national, sub-côtes XX10, etc).
- Auto-pick : MF si key dispo, sinon opendatasoft. Override VIGILANCE_PROVIDER.
- Fallback opendatasoft si MF échoue (cache key différent pour pas polluer).
- Devrait fixer le lag de plusieurs heures observé sur Opendatasoft.
2. lib/climato.ts : fetch latest + previous (~4 MB compressé), garder 365j max
en cache 24h. Permet l'onglet "1 an" sur la page dept.
3. TemperatureChartInteractive : onglet "1 an" (apparaît si > 30j dispos),
série days365 + normales365 sérialisés au SSR.
4. Base.astro : logo header et footer utilisent <img src="/favicon.svg">
au lieu d'un emoji 🌡️.
5. index.astro : wrapper map repassé à container-tight (max-w-5xl) — la version
1400px était trop grande, on revient à la largeur du reste du site.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
3 fixes en un :
1. lib/observations.ts : dedupe par validity_time sur le parse hourly.
L API MF SYNOP retourne chaque obs en doublon exact sur les ranges
multi-step (bug gateway WSO2). Constaté 7/8 paires identiques sur 24h.
2. Normales 1991-2020 passées de mensuelles à journalières (lissées 7j).
- scripts/build-normales.mjs : agrégation par day-of-year (1..366)
avec moving average ±3j pour stabiliser le bruit jour-à-jour.
- src/data/normales.json : 2.28 MB (vs 78 KB), 96 × 366 entrées.
- lib/normales.ts : normaleForDay/Date + normalesForRange, computeAnomaly
compare maintenant chaque jour observé à SA normale (pas à la moyenne
du mois) → bien plus précis sur les jours de transition mensuelle.
- TemperatureChartInteractive : overlay normales en COURBE qui suit
la saison (7j/30j) au lieu d une ligne horizontale unique.
24h reste ligne horizontale (normale du jour courant).
- Tooltip 7j/30j ajoute "↳ normale TX X°C (+Y)" pour montrer l écart
par point.
3. Carte sur la home libérée du container-tight (max-w-5xl = 1024px) :
wrapper dédié max-w-[1400px] → carte ~37% plus grande sur PC large.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
- scripts/build-normales.mjs : agrégation TN/TX mensuelles par dept sur
la période WMO 1991-2020 depuis les fichiers Q_<DEPT>_previous-1950-2024.
Output src/data/normales.json (78 KB, committé). Run annuel max.
- Corse : Météo France utilise le code historique "20" (avant split 2A/2B
en 1976), donc 2A et 2B partagent la même normale issue de Q_20_*.
- src/lib/normales.ts : computeAnomaly() qui moyenne TX/TN des 7 derniers
jours, compare à la normale du mois, calcule l'écart en °C et en σ,
catégorise (normal / warm / cool / anomaly_warm / anomaly_cool /
extreme_warm / extreme_cool / unknown).
- src/components/AnomalyBadge.astro : badge coloré (vert/jaune/orange/rouge)
visible sur /departement/[code] juste au-dessus du graphe T°.
Différencie "il fait chaud" de "il fait anormalement chaud pour ce mois".
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Quick wins :
- public/og-image.png (1200x630, via sharp depuis le SVG, build via pnpm build)
SVG ne fonctionne pas pour Open Graph (Slack/Discord/X/FB).
- @sentry/astro intégré conditionnellement (skip si SENTRY_DSN absent → no-op).
GIT_COMMIT_SHA en var pour le release tag dans GlitchTip si voulu.
- /api/vigilance/dept/[code] : JSON enrichi (phenomenon label + color name)
pour J et J1, CORS *, Cache-Control 5min. 404 si code unknown.
- JSON-LD enrichi : @graph WebSite + Service avec isBasedOn Dataset + license LOv2.
- Lien retour vigilance.meteofrance.fr visible sous la carte.
DROM (97x / 976) :
- 5 entrées ajoutées dans departements.ts (région "DROM").
- /departement/[code] DROM : bannière "Vigilance Outre-mer non couverte par
cette source open data" + bouton vers vigilance.meteofrance.fr.
- Home : ligne sous la carte listant les 5 DROM + lien retour.
- L'API /api/vigilance/dept/<DROM> retourne quand même un JSON 200 (arrays vides).
Registre canicule :
- Page /conseils/registre-canicule : qui, quoi, comment s'inscrire au CCAS.
- Numéro vert 0 800 06 66 66.
- Bannière mise en avant en haut de /conseils.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
Astro 5 SSR + ioredis cache Valkey, déployable sur shared-net.
- Vigilance temps réel via Opendatasoft (no-auth, LOv2)
- Carte SVG des 96 départements (gregoiredavid/france-geojson)
- Climato T° 30j par dept (CSV.GZ Météo France, cache 24h)
- Conseils officiels par phénomène (7 types Vigilance)
- /api/health (UptimeRobot) + /api/vigilance (JSON public CORS *)
- Dockerfile multi-stage, CI Forgejo deploy.yml (pattern Reteno)
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>