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:
@@ -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")
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user