feat: mensajes progresivos en Matrix con ProgressReporter

Implementa la Fase 2 del issue 0036: mensajes de progreso en tiempo real
que muestran al usuario que herramientas esta usando el agente claude-code.

- SendMarkdownGetID en shell/matrix/client.go: envia mensaje y retorna
  el event ID para editarlo despues
- EditMessage en shell/matrix/client.go: edita un mensaje existente
  usando m.replace (m.relates_to con rel_type=m.replace)
- ProgressReporter en shell/effects/progress.go (NEW): recibe streaming
  events y actualiza un mensaje unico en Matrix mostrando el progreso
  (e.g. "🔧 Bash: ls -la" → "🔧 Read: file.go" → " Completado")
- Rate limiter integrado: max 1 edit/segundo para no saturar el homeserver
- Conectado en devagents/handler.go: cuando provider=claude-code y
  streaming+show_tool_progress habilitados, crea ProgressReporter y
  pasa StreamFunc al CompletionRequest
- MatrixSender interface actualizada con los nuevos metodos
- 10 tests nuevos para ProgressReporter, todos los existentes pasan

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-04-09 22:58:03 +00:00
parent 1bdf9344a2
commit 45bd258be1
8 changed files with 482 additions and 8 deletions
+19 -5
View File
@@ -82,6 +82,20 @@ func (s *spyMatrixSender) SendMarkdown(_ context.Context, roomID, markdown strin
return nil
}
func (s *spyMatrixSender) SendMarkdownGetID(_ context.Context, roomID, markdown string) (string, error) {
s.mu.Lock()
defer s.mu.Unlock()
s.messages = append(s.messages, sentMessage{roomID: roomID, text: markdown})
return "$spy_event_id", nil
}
func (s *spyMatrixSender) EditMessage(_ context.Context, roomID, originalEventID, markdown string) error {
s.mu.Lock()
defer s.mu.Unlock()
s.messages = append(s.messages, sentMessage{roomID: roomID, text: markdown, inReplyTo: originalEventID})
return nil
}
func (s *spyMatrixSender) SendReplyMarkdown(_ context.Context, roomID, inReplyTo, markdown string) error {
s.mu.Lock()
defer s.mu.Unlock()
@@ -590,7 +604,7 @@ func TestRunLLM_ToolCallExecutesAndReturns(t *testing.T) {
IsDirectMsg: true,
}
reply, err := a.runLLM(context.Background(), msgCtx, "!room:example.com")
reply, err := a.runLLM(context.Background(), msgCtx, "!room:example.com", nil)
if err != nil {
t.Fatalf("runLLM error: %v", err)
}
@@ -655,7 +669,7 @@ func TestRunLLM_ToolCallFailsPassesErrorToLLM(t *testing.T) {
Content: "do something",
}
reply, err := a.runLLM(context.Background(), msgCtx, "!room:example.com")
reply, err := a.runLLM(context.Background(), msgCtx, "!room:example.com", nil)
if err != nil {
t.Fatalf("runLLM error: %v", err)
}
@@ -716,7 +730,7 @@ func TestRunLLM_MaxIterationsRespected(t *testing.T) {
Content: "loop please",
}
reply, err := a.runLLM(context.Background(), msgCtx, "!room:example.com")
reply, err := a.runLLM(context.Background(), msgCtx, "!room:example.com", nil)
if err != nil {
t.Fatalf("runLLM error: %v", err)
}
@@ -776,7 +790,7 @@ func TestRunLLM_RBACDeniesToolCall(t *testing.T) {
Content: "use restricted tool",
}
reply, err := a.runLLM(context.Background(), msgCtx, "!room:example.com")
reply, err := a.runLLM(context.Background(), msgCtx, "!room:example.com", nil)
if err != nil {
t.Fatalf("runLLM error: %v", err)
}
@@ -819,7 +833,7 @@ func TestRunLLM_LLMError(t *testing.T) {
Content: "hello",
}
_, err := a.runLLM(context.Background(), msgCtx, "!room:example.com")
_, err := a.runLLM(context.Background(), msgCtx, "!room:example.com", nil)
if err == nil {
t.Fatal("expected error from LLM, got nil")
}