chore: add CLAUDE.md agent context

This commit is contained in:
Patrick Schurig 2026-04-04 17:35:57 +02:00
parent 0bf5003992
commit 1278da2a04

196
CLAUDE.md
View File

@ -6,10 +6,13 @@ This file provides guidance to Claude Code (claude.ai/code) when working with co
Dossier parent de tous les projets : `/home/etm/Projects/`
- **`etm_powersync_app/`** (ce projet) — application Flutter HEMS, client mobile
- **`etm_powersync_app/`** (ce projet) — application Flutter HEMS, interface utilisateur
- **`etm-nymea/`** — serveur nymea et plugins associés (C++/Qt), le backend
Ce projet est le **client Flutter** qui communique avec le serveur nymea via JSON-RPC 2.0.
Il n'a aucun accès direct aux plugins — tout passe par les APIs JSON-RPC.
Lire aussi le `CLAUDE.md` du dossier parent (`/home/etm/Projects/CLAUDE.md`) pour le contexte global.
---
@ -23,6 +26,20 @@ flutter analyze # analyse statique Dart
flutter test # lancer les tests
```
## Stack technique
| Élément | Valeur |
|---|---|
| Flutter SDK | ^3.11.0 |
| State management | `provider ^6.1.2``NymeaService` (ChangeNotifier unique) |
| Navigation | `go_router ^14.6.3` — ShellRoute + 25+ routes |
| Transport | WebSocket port 4444 / TCP brut port 2222 |
| Graphiques | `fl_chart ^0.70.2` |
| Persistance locale | `shared_preferences ^2.3.2` |
| Sécurité installateur | `crypto ^3.0.6` (PIN SHA-256, lock 30s, auto-lock 10 min) |
---
## Architecture
### State management — un seul Provider
@ -50,7 +67,7 @@ flutter test # lancer les tests
Les requêtes utilisent des IDs auto-incrémentés stockés dans `_pendingRequests: Map<int, Completer>`. Timeout global : 15 secondes. Les notifications push nymea (pas de champ `id`, ont un champ `notification`) sont dispatchées dans `_handleNotification()`.
**Convention nymea énergie importante** : `currentPowerProduction` est **négatif** (producteur = négatif). Le service convertit en positif dans `_parsePowerBalance()`.
**Convention nymea énergie importante** : `currentPowerProduction` est **négatif** (producteur = négatif). Le service convertit en positif dans `_parsePowerBalance()`. Puissances en **W**, énergies journalières en **Wh**, totaux API nymea en **kWh** (× 1000 dans le service). Timestamps : **secondes** dans toutes les APIs energy.
### Mode simulation
@ -65,7 +82,160 @@ Les requêtes utilisent des IDs auto-incrémentés stockés dans `_pendingReques
### Thème
`AppTheme` (`lib/theme/app_theme.dart`) expose des constantes `Color` nommées : `primaryGreen`, `solarYellow`, `gridGray`, `homeBlue`, `batteryGreen`, `boostRed`, `pvGreen`, `minPvBlue`, etc. Toujours utiliser ces constantes plutôt que des valeurs hex brutes.
`AppTheme` (`lib/theme/app_theme.dart`) expose des constantes `Color` nommées. Toujours utiliser ces constantes plutôt que des valeurs hex brutes :
```dart
primaryGreen // principal
solarYellow // production PV
gridGray // réseau
homeBlue // consommation maison
batteryGreen // batterie
boostRed // mode boost
pvGreen // surplus PV
minPvBlue // mode minPV
accentTeal // accent
```
---
## 🔴 Bug critique — priorité absolue
`EVChargingCard` appelle `Energy.SetChargingMode` — cette méthode appartient à l'experience plugin et concerne le comptage du rootmeter, **pas le pilotage EV**. Le SmartChargingManager n'est jamais déclenché depuis l'app dans l'état actuel.
```dart
// ❌ FAUX — ne déclenche pas le SmartChargingManager
nymeaService.call('Energy.SetChargingMode', {'mode': 'pv'});
// ✅ CORRECT — appelle vraiment le plugin ETM
nymeaService.call('EnergyPlugin.SetChargingInfo', {
'chargingInfo': {
'evChargerId': chargerId,
'mode': 'Eco', // Normal | Eco | EcoWithTargetTime
'targetSoc': 80,
'endTime': null,
}
});
```
---
## APIs JSON-RPC consommées
### `Energy.*` (nymea-experience-plugin-energy)
| Méthode | Usage |
|---|---|
| `GetPowerBalance` | Dashboard temps réel |
| `GetPowerBalanceLogs(sampleRate, from, to)` | Historique — envoyer les bons `from/to` pour les 90j |
| `GetThingPowerLogs(sampleRate, thingIds[], from, to)` | Historique par device |
| `SetRootMeter(rootMeterThingId)` | Config installateur |
**Notifications push :** `PowerBalanceChanged`, `PowerBalanceLogEntryAdded`, `ThingPowerLogEntryAdded`, `RootMeterChanged`
### `EnergyPlugin.*` (powersync-energy-plugin-etm)
| Méthode | Usage |
|---|---|
| `GetChargingInfos(o:evChargerId)` | Lire config recharge EV |
| `SetChargingInfo(chargingInfo)` | **Piloter le smart charging EV** (Eco / EcoWithTargetTime / Normal) |
| `GetChargingSchedules(o:evChargerId)` | Afficher le planning EV calculé |
| `GetAvailableSpotMarketProviders()` | Liste des fournisseurs tarifs disponibles |
| `SetSpotMarketConfiguration(enabled, providerId)` | Activer/choisir fournisseur spot market |
| `GetSpotMarketScoreEntries(o:date)` | Cotations horaires aWATTar |
| `SetPhasePowerLimit(Uint)` | Config installateur — protection surcharge (A/phase) |
| `SetAcquisitionTolerance(Double)` | Config seuil surplus déclenchant la recharge |
| `SetBatteryLevelConsideration(Double)` | Facteur batterie dans le calcul surplus |
**Notifications push :** `ChargingInfoAdded/Removed/Changed`, `ChargingSchedulesChanged`, `SpotMarketConfigurationChanged`, `SpotMarketScoreEntriesChanged`, `PhasePowerLimitChanged`
### `AirConditioning.*` (nymea-experience-plugin-airconditioning)
| Méthode | Usage |
|---|---|
| `GetZones` | Afficher les zones PAC/thermostat |
| `AddZone` / `RemoveZone` | Gestion des zones |
| `SetZoneName` / `SetZoneThings` | Configuration zone |
| `SetZoneStandbySetpoint` | Consigne hors-horaire |
| `SetZoneSetpointOverride` | Pilotage manuel zone |
| `SetZoneWeekSchedule` | Planning 7 jours |
**Notifications push :** `ZoneAdded`, `ZoneRemoved`, `ZoneChanged`
### `Integrations.*` / `Rules.*` / `Logging.*`
- `GetThings`, `GetThingClasses`, `ExecuteAction`, `SetStateValue`, `SetThingSettings`, `GetStateValue` — ✅ déjà implémentés
- `DiscoverThings`, `AddThing`, `RemoveThing`, `EditThing` — ✅ déjà implémentés
- `GetRules` — ✅ lecture seule déjà implémentée
- `AddRule`, `RemoveRule`, `EditRule` — ❌ à implémenter (UI automatisations Community)
- `Logging.GetLogEntries` — ✅ implémenté (historique SOC batterie, température)
---
## État des écrans
| Écran | État | Action requise |
|---|---|---|
| Dashboard (Sankey + gains + EV card) | ✅ | Icône notifications à brancher |
| EnergyScreen (4 onglets + charts) | ✅ | Ajouter sélecteur plage 90j |
| ThingsScreen + ThingDetailScreen | ✅ | — |
| FavoritesScreen | ✅ | **Persister dans SharedPreferences** |
| InstallerMode (PIN SHA-256) | ✅ | — |
| RoleConfigFlow wizard | ⚠️ Stub | Brancher Step 3 sur les vrais RPC |
| TariffScreen | ⚠️ Stub | Brancher sur `EnergyPlugin.SetSpotMarketConfiguration` |
| SchedulerScreen | ⚠️ Stub | `setStrategy()` / `forceRecalc()` = fonctions vides |
| TimelineScreen | ⚠️ Stub | Données simulées → brancher scheduler réel |
| AirConditioning zones | ❌ Absent | À créer (tier Auto) |
| DeveloperScreen | ❌ Vide | À créer |
| AboutScreen | ❌ Vide | À créer |
---
## Persistance
| Donnée | Actuellement | À faire |
|---|---|---|
| Adresse serveur, PIN, préférences UI | ✅ SharedPreferences | — |
| `RoleAssignments` (configuration EMS) | ❌ En mémoire | Persister SharedPreferences |
| `FavoriteWidgets` | ❌ En mémoire | Persister SharedPreferences |
| `TariffConfig` / `HcHpConfig` | ❌ En mémoire | Persister SharedPreferences |
| `SchedulerConfig` | ❌ En mémoire | Persister SharedPreferences |
---
## Feature gating par tier
Le tier actif est lu depuis `/etc/powersync/tier.conf` via un futur RPC.
En attendant, utiliser `TierProvider` (classe à créer) avec valeur par défaut `community`.
Utiliser `pro_lock_badge.dart` (déjà présent) pour verrouiller visuellement les features non disponibles. **Ne jamais hardcoder le tier** — toujours passer par `TierProvider` :
```dart
// Toujours via TierProvider — jamais de valeur hardcodée
if (tierProvider.tier >= Tier.auto) {
// afficher feature Auto
}
```
| Feature | Community | Auto | Predict AI |
|---|---|---|---|
| Dashboard temps réel | ✅ | ✅ | ✅ |
| Historique journée en cours | ✅ | ✅ | ✅ |
| Config EV (`EnergyPlugin.SetChargingInfo`) | ✅ | ✅ | ✅ |
| Tarif HP/HC statique | ✅ | ✅ | ✅ |
| UI automatisations (`Rules.*`) | ✅ | ✅ | ✅ |
| Historique 90 jours | 🔒 | ✅ | ✅ |
| Wizard onboarding guidé | 🔒 | ✅ | ✅ |
| Zones PAC/ECS (`AirConditioning.*`) | 🔒 | ✅ | ✅ |
| Prévision solaire Open-Meteo | 🔒 | ✅ | ✅ |
| Notifications d'anomalies | 🔒 | ✅ | ✅ |
| Accès distant sécurisé | 🔒 | ✅ | ✅ |
| Tarifs dynamiques aWATTar / spot market | 🔒 | 🔒 | ✅ |
| ENTSO-E / Tibber | 🔒 | 🔒 | ✅ |
| Accès fonctionnalités beta | 🔒 | 🔒 | ✅ |
Flavors Android/iOS à configurer : `com.etm-powersync.community` / `.auto` / `.predictai`
---
## Conventions
@ -74,20 +244,16 @@ Les requêtes utilisent des IDs auto-incrémentés stockés dans `_pendingReques
- Les écrans utilisent `Consumer<NymeaService>` ou `context.watch` — ne jamais dupliquer l'état du service dans un `State`
- **Unités** : Watts (W) pour les puissances instantanées, Wh pour les énergies journalières. L'API nymea retourne des totaux en kWh que le service multiplie par 1000.
## Méthodes nymea principales
---
| Méthode | Rôle |
|---------|------|
| `JSONRPC.Hello` | Handshake (WebSocket uniquement — envoyé par le client) |
| `Integrations.GetThings` | Charger les devices configurés |
| `Integrations.GetThingClasses` | Charger les définitions de classes (filtre `thingClassIds` supporté) |
| `Integrations.ExecuteAction` | Exécuter une action sur un thing |
| `Integrations.GetStateValue` | Lire une valeur d'état |
| `Energy.GetPowerBalance` | Snapshot du flux énergétique actuel |
| `Energy.GetEnergyLogs` | Historique énergie (échantillons horaires) |
| `Energy.SetChargingMode` | Mode borne EV : `pv`, `minpv`, `boost` |
## Règles de modification
Notifications push gérées : `Energy.PowerBalanceChanged`, `Energy.RootMeterChanged`, `Integrations.StateChanged`, `Integrations.ThingAdded`, `Integrations.ThingRemoved`.
- Tout nouvel écran → valider la maquette avec Patrick avant de coder
- Tout nouvel appel RPC → vérifier dans `docs/nymea_api.md` que la méthode existe
- Ne **jamais** appeler `Energy.SetChargingMode` pour piloter un EV
- Toujours tester la persistance : killer l'app et vérifier que les données survivent au redémarrage
---
## Documentation API