feat(lea): bulle paused_need_help interactive — asset démo majeur

Quand Léa bascule en pause supervisée (event 'lea:paused'), affichage d'une
bulle dédiée dans ChatWindow avec encadré orangé, raison de la pause, et deux
boutons Continuer/Annuler. C'est le moment qui incarne la différence RPA classique
vs Léa devant Carvella : Léa SAIT qu'elle ne sait pas et demande de l'aide.

Architecture (canal SocketIO bidirectionnel, pas de nouvel endpoint streaming) :

  ChatWindow ──[lea:replay_resume]──> agent_chat ──POST /resume──> streaming
  ChatWindow ──[lea:replay_abort ]──> agent_chat (running=False local)

Composants ajoutés :
- agent_chat/app.py : handlers 'lea:replay_resume' / 'lea:replay_abort' +
  acks 'lea:resume_acked' / 'lea:abort_acked' pour feedback côté client
- network/feedback_bus.py : méthodes resume_replay() / abort_replay() avec
  helper _safe_emit (silencieux + retourne bool succès)
- ui/chat_window.py : palette PAUSED_*, _add_paused_bubble(),
  _render_paused_bubble(), _close_active_paused_bubble() (auto-fermeture
  sur lea:resumed/done), _on_paused_resume/abort

8 nouveaux tests pytest (4 handlers serveur + 4 méthodes client).
Total branche : 29/29 verts.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Dom
2026-04-28 10:08:32 +02:00
parent 6154423a91
commit 2af3bc3b93
5 changed files with 314 additions and 0 deletions

View File

@@ -123,3 +123,42 @@ def test_stop_silenced_on_disconnect_error():
with patch.object(bus._sio, 'disconnect', side_effect=RuntimeError("boom")):
bus._sio.connected = True
bus.stop() # ne doit pas raise
# ----------------------------------------------------------------------
# J3.5 — Actions utilisateur (resume_replay / abort_replay)
# ----------------------------------------------------------------------
def test_resume_replay_emits_when_connected():
bus = FeedbackBusClient("http://localhost:5004")
bus._sio.connected = True
with patch.object(bus._sio, 'emit') as mock_emit:
ok = bus.resume_replay("rep_abc")
assert ok is True
mock_emit.assert_called_once_with("lea:replay_resume", {"replay_id": "rep_abc"})
def test_resume_replay_returns_false_when_disconnected():
bus = FeedbackBusClient("http://localhost:5004")
# _sio.connected reste False par défaut
with patch.object(bus._sio, 'emit') as mock_emit:
ok = bus.resume_replay("rep_abc")
assert ok is False
mock_emit.assert_not_called()
def test_abort_replay_emits_when_connected():
bus = FeedbackBusClient("http://localhost:5004")
bus._sio.connected = True
with patch.object(bus._sio, 'emit') as mock_emit:
ok = bus.abort_replay("rep_xyz")
assert ok is True
mock_emit.assert_called_once_with("lea:replay_abort", {"replay_id": "rep_xyz"})
def test_safe_emit_silenced_on_error():
bus = FeedbackBusClient("http://localhost:5004")
bus._sio.connected = True
with patch.object(bus._sio, 'emit', side_effect=RuntimeError("boom")):
ok = bus.resume_replay("rep_abc")
assert ok is False # erreur avalée silencieusement