gilles
fa73ab07b0
feat(events): timeline d'événements machine (tâche 5 backlog)
...
- listMachineEvents (machine_events, 30 derniers, desc) + route GET /machines/:id/events
- api machineEvents ; section repliable « Timeline » dans le panneau détail
(badge sévérité + horodatage), exploite les events déjà enregistrés par recordEvent
tsc 0 · 118 tests · build OK.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 19:36:06 +02:00
gilles
a93a43e1c8
feat(messages): extraction des messages importants APT (tâche 5 backlog)
...
- extractImportantMessages (TDD) : E:/dpkg error → error, W:/GPG → warning,
déprecations/EOL → future_major_change ; nettoyage des secrets dans les URLs
- recordImportantMessages : dédup par (machine, source, message) non acquitté →
maj lastSeenAt, sinon insert (firstSeen/lastSeen) dans important_messages
- branché dans refreshMachine (sortie APT) avec snapshotId
- routes GET /machines/:id/messages + POST .../:msgId/ack
- UI : carte « Messages importants » (badge sévérité + ack) dans le panneau détail
tsc 0 · 118 tests · build OK.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 19:30:07 +02:00
gilles
ff9cfaa9e1
feat(scheduler): automatisations planifiées (cron) — tâche 5
...
- table schedules (migration 0007) + service scheduler (croner) : CRUD,
runSchedule avec scope (all/liste), pool de concurrence et verrou par machine,
mapping actions → refresh/metrics/docker_scan ; reloadSchedules au boot
- worker = reloadSchedules (remplace le refresh 30 min en dur)
- routes /api/schedules (CRUD + :id/run) ; cron invalide rejeté (validation croner)
- UI Paramètres : onglet « Automatisations » (liste, activer/lancer/supprimer, création)
tsc 0 · 113 tests · build OK · boot OK (migration 0007, CRUD vérifié).
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 19:25:44 +02:00
gilles
1530409d3b
feat(post-install): identity_network reboote et rebascule l'IP en BDD (tâche 4)
...
- updateMachine accepte name/hostname/port (correction d'identité réseau)
- rebootAndRebind : reboot sur l'ancienne connexion → attente du retour sur la
NOUVELLE IP (verifyReboot avec host cible) → maj BDD (hostname + nom) si OK.
Sécurité : BDD inchangée si la machine ne revient pas (récupération console/backups)
- execute post_install : si identity_network + reboot coché + succès, déclenche
rebootAndRebind et joint le RebootResult ; statut error si reconnexion échoue
tsc 0 · 113 tests · build OK.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 19:12:39 +02:00
gilles
3ea2e66359
fix(post-install): identity_network cadré Debian/ifupdown (VM) avec précheck
...
- précheck en tête : refuse proprement si OS != debian (os_not_supported),
si netplan présent (unsupported_network_manager) ou si /etc/network/interfaces
absent (ifupdown_not_found) — au lieu d'écrire une conf inopérante
- manifeste : label « (Debian/VM) » + description précisant la cible ifupdown
et l'application au reboot
Validé en réel sur Debian VM (ens18) : strophe DHCP commentée + drop-in statique.
sh -n OK · tsc 0 · 113 tests.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 19:03:28 +02:00
gilles
d1b0290e3b
feat(apt): analyse des dépôts APT (lecture seule) (tâche 4)
...
- template repositories (deb lines + deb822), non destructif
- analyzeRepositories (TDD) : composants, repos, détection Proxmox
enterprise/no-subscription, warnings (pve_enterprise_without_subscription,
pve_repo_missing) + notes Debian/Ubuntu composants manquants
- route POST /machines/:id/apt-repositories ; api analyzeRepositories
- popup config : bloc « Dépôts APT » (composants + warnings + notes)
Analyse uniquement (modification = action validée séparée, future). tsc 0 · 113 tests · build OK.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 18:41:11 +02:00
gilles
e3e824185f
feat(probe): sonde enrichie CPU/RAM/disques + recommandations de profils (tâche 4)
...
- template machine-probe : lscpu Model name + nproc, MemTotal, lsblk disques
- parseProbe étendu (cpuModel/cpuCores/memoryBytes/disks) + buildRecommendations
(KVM/QEMU → vm_guest_tools) ; tests TDD
- runProbe persiste cpu/mem/disks dans machine_hardware ; /probe renvoie recommendations
- popup Sonde affiche cpu/ram/disks + profils conseillés
tsc 0 · 110 tests · build OK.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 18:36:49 +02:00
gilles
c390addadb
feat(metrics): machine_metrics_simple — CPU/RAM/disque live par machine (tâche 4)
...
- template machine-metrics (loadavg/nproc, /proc/meminfo, df -B1) non destructif
- parseMetrics (TDD) → cpu load/cores, mémoire kB→B + %, filesystems, warnings >=90%
- collectMetrics (SSH léger) persiste machine_metrics_latest ; getLatestMetrics (sans SSH)
- routes GET /machines/:id/metrics + POST /metrics/collect ; api latestMetrics/collectMetrics
- section Hardware : bloc métriques live (CPU/RAM/disques + alertes) + bouton Collecter
→ comble le gap « Health » de la tâche 3
tsc 0 · 108 tests · build OK.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 17:01:45 +02:00
gilles
58abebf687
feat(ui): ajout machine OS/type, section Hardware, identité app (tâche 3)
...
- AddMachineModal : sélecteurs OS + Type machine ; createMachine accepte
osFamily/machineKind (manuel prioritaire, "Autre/auto" → détection os-release)
- section Hardware sur la tuile + panneau détail : os/type/virt/arch/gpu/réseau
depuis machine_hardware (sonde) via GET /machines/:id/hardware
- identité : favicon.svg (serveur + LED Gruvbox), favicon.ico, apple-touch-icon,
PWA 192/512, site.webmanifest ; liens + theme-color dans index.html
tsc 0 · 104 tests · build OK.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 14:07:46 +02:00
gilles
3b16fdd52a
feat(post-install): catalogue de profils — paquets, Docker officiel, partages, VM tools (tâche 2 SJ-9)
...
- mécanisme presetVars (variables fixes injectées au rendu, surchargées par le formulaire)
- 6 profils : base_tools / network_tools / dev_git (listes de paquets, low),
docker_official (dépôt officiel Debian, confirmation), sharing (Samba/NFS/mDNS, confirmation),
vm_guest_tools (qemu/vmware)
- 4 templates custom (install-package-groups, docker-official-debian, sharing, vm-guest-tools)
émettant PKG_INSTALLED/SERVICE_ENABLED/ERR → réutilise buildPostInstallResult
- l'UI post-install générique les expose automatiquement (manifeste → formulaire → run)
tsc 0 · 104 tests · build OK · boot OK (8 profils servis). Clôt le volet moteur tâche 2.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 12:40:20 +02:00
gilles
e6f4ae470b
feat(post-install): moteur de profils + bootstrap + identité/réseau (tâche 2 SJ-8)
...
- templates custom/bootstrap-root + identity-network (sortie structurée parsable,
sauvegardes, échec contrôlé, jamais de coupure réseau sans reconnexion)
- postInstall: registre de manifestes (champs typés + defaults/defaultFrom),
validateProfileValues + maskSecretValues + buildPostInstallResult (TDD),
renderProfile/previewProfile (masquage secrets), runPostInstall (SSH)
- execute: RunActionOpts.profileId/values + branche post_install (bloc postInstall)
- action_requests: post_install accepté, payload profileId/values transmis à approve
- routes: GET /profiles, POST .../preview (script masqué + validation),
POST .../run (action_request si requiresConfirmation, sinon direct)
Champs = formulaire (pas de question SSH interactive) ; secrets jamais sérialisés ;
identity_network exige confirmation. tsc 0 · 101 tests · build OK · boot OK.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 08:02:32 +02:00
gilles
2b684da9cd
feat(api): profil machine éditable, sonde, et réglages globaux apt-cacher-ng
...
- machines : updateMachine (PATCH /machines/:id) + POST /machines/:id/probe
(sonde synchrone → faits + proposition de correction) ; MachineView expose
machineKind/virtualization ; CreateMachineInput accepte aptProxyMode persistent
- app_settings (clé/valeur, migration 0006) + service appSettings :
défaut apt-cacher-ng (mode + url) ; applyProxyToAllMachines
- routes /settings : GET, PUT /apt-proxy, POST /apt-proxy/apply-all
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 07:53:47 +02:00
gilles
bafb085995
feat(os): profils Proxmox/RPi + machine_probe + proxy persistent (tâche 2 SJ-7)
...
- templates proxmox/ (update-analyze: dépôts PVE ; full-upgrade) et raspbian/
(update-analyze: espace disque ; full-upgrade)
- execute résout les actions APT par profil OS (resolveTemplate) → proxmox/
raspbian si dispo, sinon fallback apt/ (non-régression debian/ubuntu vérifiée)
- machine_probe (lecture seule) : template + parseProbe/proposeCorrections (TDD)
→ propose os_family/machine_kind/virtualization, persiste machine_hardware,
n'applique jamais auto ; branche execute + allowlist route
- apt_proxy_persistent : ActionType + template idempotent (/etc/apt/apt.conf.d/
01proxy, backup) + TemplateVars.aptProxyUrl + allowlist route
tsc 0 · 95 tests · build OK · résolution OS vérifiée.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 07:14:43 +02:00
gilles
2c15b8c06b
feat(docker): routes de gestion des stacks (settings/roots/scan/list/enable)
...
Rend le flux Docker déclenchable via l'API (prérequis SJ-5/SJ-6) :
- GET /machines/:id/docker/settings — settings + racines Compose
- POST /machines/:id/docker/roots — déclare/active les racines à scanner
- POST /machines/:id/docker/scan — scan passif (background, WS)
- GET /machines/:id/docker/stacks — liste stacks + services
- PATCH /machines/:id/docker/stacks/:stackId — cycle candidate→enabled→ignored
dockerScan: getDockerSettings, listStacks, setStackStatus. Les actions
pull-check/apply/down restent réservées aux stacks enabled.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 06:24:43 +02:00
gilles
47fe952240
feat(settings): backup/restore de la base de données (amelioration #4 )
...
- service dbBackup : createBackup (VACUUM INTO → archive .db cohérente),
validateSqlite (header + integrity_check + schéma), prepareRestore
(sauvegarde de sécurité auto + dépôt <db>.incoming)
- swap hors-ligne au démarrage (db/client.ts) : aucune corruption d'une base
ouverte ; restauration appliquée au redémarrage
- routes GET /system/db/info|backup, POST /system/db/restore
- lib api : dbInfo / dbBackup (download navigateur) / dbRestore (upload)
- SettingsModal : onglet « Base de données » (taille, télécharger, restaurer
avec confirmation Popup), icônes database/upload, styles DS variables only
Testé end-to-end : backup 184 Ko valide, restore + safety .bak + swap au boot,
fichier invalide rejeté. tsc 0 erreur · 91 tests · build OK.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 06:13:03 +02:00
gilles
edb22a59c7
feat(docker): apply/prune/down + socle action_requests (tâche 2 SJ-6)
...
- migration 0005 : tables docker_image_events + action_requests
- templates apply-compose (up -d --remove-orphans), prune-images (safe/agressif),
down-compose (sans volumes/rmi)
- dockerApply: parsers TDD (apply recreated/running/exited, prune images+bytes,
down removed, parseHumanBytes) + orchestration applyStack/pruneImages/downStack
réservée aux stacks enabled, insère docker_image_events
- actionRequests: create/approve/reject/list — actions destructives validées
explicitement (Hermes propose, opérateur approuve, run en arrière-plan) ;
hors API directe (POST /:id/actions reste passif uniquement)
- routes /machines/:id/action-requests + /action-requests/:id[/approve|/reject]
- execute: RunActionOpts.aggressive, branches apply/prune/down, helper
archiveExecution mutualisant le boilerplate d'archivage
tsc 0 erreur · 91 tests · build OK · boot OK (migrations 0000→0005).
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-06 06:05:59 +02:00
gilles
b1c81ba518
feat(docker): pull-check + comparaison déterministe par stack (tâche 2 SJ-5)
...
- template docker/pull-check.sh.tpl (pull sans up, inspect before/after)
- dockerPull: parseDockerPullCheck + buildDockerPullResult (TDD) — compare
image id/digest/label OCI → services up_to_date|updates_available|error,
changes operation=pulled ; erreurs registry nettoyées (URL/token/password)
- dockerDedupKey (digests prioritaires, fallback image ids) + DockerImageChange.dedupKey
- pullCheckStack: SSH + upsert docker_stack_services, refuse stack non enabled,
refresh Docker séparé (hors refreshMachine, pas de pull auto)
- execute: runAction(opts.stackId), branche docker_pull_check, injection stackDir
(corrige docker_inspect_current) ; route: allowlist Docker passifs + pull_check,
destructives toujours hors API jusqu'à action_requests (SJ-6)
Pas de migration (schéma SJ-4 suffisant). tsc 0 erreur · 85 tests · build OK.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-05 21:02:38 +02:00
gilles
2af8e74079
feat(docker): scan/inspect passifs des stacks Compose (tâche 2 SJ-4)
...
- 4 tables Docker (settings/compose_roots/compose_stacks/stack_services)
+ migration 0004 (timestamps journal monotones)
- templates docker/scan-compose + inspect-compose ; renderTemplate bascule
sur délimiteurs <% %> pour les templates docker/ afin de préserver les
Go-templates {{.ID}} intacts
- dockerScan: parseDockerScan (TDD) + scanDockerStacks (persiste stacks
candidats, complète la détection par labels)
- action docker_scan branchée dans execute (route dédiée, archivage report/log)
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-05 20:54:52 +02:00
gilles
434a149f1f
fix(execute): refresh snapshot après apt upgrade/full-upgrade (amelioration #3 )
...
Après une action APT appliquée avec succès, relance refreshMachine pour
que la webui reflète l'état réel des paquets. Échec de refresh = event
warning non bloquant (post_action_refresh_failed).
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-05 20:54:38 +02:00
gilles
08919752e3
feat: socle BDD (tâche 1.9 Phase 1-2) + moteur APT (tâche 2 SJ-0→3) + WIP capabilities/auth/Rust
...
Checkpoint multi-chantiers (arbre vert : tsc 0 erreur, 70 tests, build OK).
- tâche 1.9 Phase 1 : schéma socle (machine_state/events/reports/raw_artifacts/
hardware/metrics + colonnes étendues) + wiring refresh/execute. Migration 0002.
- tâche 1.9 Phase 2 : machine_credentials + machine_host_keys (non destructif,
dual-read + backfill). Migration 0003. Fix séquence journal de migration.
- tâche 2 : SJ-0 (types étendus rétro-compatibles, réducteur Docker, resolveTemplate),
SJ-1 (update-analyze enrichi), SJ-2 (apply + diff dpkg + timeout inactivité SSH),
SJ-3 (reboot vérifié boot_id).
- WIP parallèle inclus : /api/capabilities, auth/apiTokens/apiClients, system metrics,
scaffold app_rust, ajustements frontend.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-05 19:50:25 +02:00
gilles
f5f361a349
fix: ORDER BY sur getLatestSnapshot + lisibilité condition EXIT
...
Suite revue batch D.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-05 04:10:56 +02:00
gilles
1fb93873ac
feat: service execute (full-upgrade/reboot -> execution + rapport archivé)
...
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-05 04:05:58 +02:00
gilles
ed3cb91cd4
feat: génération de rapport Markdown d'exécution
...
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-05 04:05:23 +02:00
gilles
0576820059
feat: service refresh (check APT -> snapshot canonique)
...
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-05 04:04:28 +02:00
gilles
3724326d81
feat: service machines (CRUD, test-connection, détection OS)
...
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-05 04:03:51 +02:00
gilles
8cce701715
feat: parser sortie apt-get -s full-upgrade -> AptPackage[]
...
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com >
2026-06-04 21:03:52 +02:00