mirror of
https://github.com/coder/coder.git
synced 2026-06-02 20:48:20 +00:00
4751416b29
**Breaking change for changelog:**
> `codersdk.Chat.last_error` now returns a structured `ChatError` object
(`{message, kind, provider, retryable, status_code, detail}`) instead of
a plain string. The chats API is experimental
(`/api/experimental/chats`), so this ships without a deprecation cycle;
consumers reading `chat.last_error` as a string must update to read
`chat.last_error.message`. SDK/generated TypeScript terminal error
payloads now use the single `ChatError` type; the live stream error
payload type is renamed from `ChatStreamError` to `ChatError`.
Persisted chat errors now carry the same provider-specific detail (kind,
provider, retryable, HTTP status, optional detail) as the live stream,
so refreshing a failed chat rehydrates with the full structured error
instead of a one-line headline.
Existing rows are migrated in place: legacy text errors are wrapped into
`{message, kind: "generic"}` so already-errored chats still render, and
rows with `last_error IS NULL` stay NULL. Internally, persisted fallback
decoding now reuses the existing `chaterror.KindGeneric` constant, with
no JSON value change.
Closes CODAGT-239
132 lines
4.3 KiB
Go
132 lines
4.3 KiB
Go
package cli //nolint:testpackage // Tests unexported chat stream helpers.
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"testing"
|
|
|
|
"github.com/google/uuid"
|
|
"github.com/stretchr/testify/require"
|
|
"golang.org/x/xerrors"
|
|
|
|
"github.com/coder/coder/v2/codersdk"
|
|
)
|
|
|
|
type chatWatchWriters struct{ stdout, stderr io.Writer }
|
|
|
|
func (w chatWatchWriters) Write(p []byte) (int, error) { return w.stdout.Write(p) }
|
|
|
|
func (w chatWatchWriters) Stderr() io.Writer {
|
|
if w.stderr != nil {
|
|
return w.stderr
|
|
}
|
|
return w.stdout
|
|
}
|
|
|
|
func consumeChatStream(eventCh <-chan codersdk.ChatStreamEvent, out io.Writer) error {
|
|
errOut := out
|
|
if writer, ok := out.(interface{ Stderr() io.Writer }); ok {
|
|
errOut = writer.Stderr()
|
|
}
|
|
|
|
printedInline := false
|
|
flush := func() error {
|
|
if !printedInline {
|
|
return nil
|
|
}
|
|
printedInline = false
|
|
_, err := fmt.Fprintln(out)
|
|
return err
|
|
}
|
|
|
|
printLine := func(dst io.Writer, format string, args ...any) error {
|
|
if err := flush(); err != nil {
|
|
return err
|
|
}
|
|
_, err := fmt.Fprintf(dst, format, args...)
|
|
return err
|
|
}
|
|
|
|
for event := range eventCh {
|
|
var err error
|
|
switch event.Type {
|
|
case codersdk.ChatStreamEventTypeMessagePart:
|
|
if part := event.MessagePart; part != nil &&
|
|
part.Part.Type == codersdk.ChatMessagePartTypeText && part.Part.Text != "" {
|
|
printedInline = true
|
|
_, err = fmt.Fprint(out, part.Part.Text)
|
|
}
|
|
case codersdk.ChatStreamEventTypeMessage:
|
|
if message := event.Message; message != nil && !printedInline {
|
|
for _, part := range message.Content {
|
|
if part.Type != codersdk.ChatMessagePartTypeText || part.Text == "" {
|
|
continue
|
|
}
|
|
printedInline = true
|
|
if _, err = fmt.Fprint(out, part.Text); err != nil {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
if err == nil {
|
|
err = flush()
|
|
}
|
|
case codersdk.ChatStreamEventTypeStatus:
|
|
if event.Status == nil {
|
|
err = flush()
|
|
break
|
|
}
|
|
err = printLine(out, "[Status: %s]\n", event.Status.Status)
|
|
case codersdk.ChatStreamEventTypeError:
|
|
if event.Error == nil {
|
|
err = flush()
|
|
break
|
|
}
|
|
err = printLine(errOut, "[Error: %s]\n", event.Error.Message)
|
|
case codersdk.ChatStreamEventTypeRetry:
|
|
if event.Retry == nil {
|
|
err = flush()
|
|
break
|
|
}
|
|
err = printLine(out, "[Retry attempt %d after error: %s]\n", event.Retry.Attempt, event.Retry.Error)
|
|
case codersdk.ChatStreamEventTypeQueueUpdate:
|
|
default:
|
|
err = printLine(out, "[Event: %s]\n", event.Type)
|
|
}
|
|
if err != nil {
|
|
return xerrors.Errorf("render chat stream event: %w", err)
|
|
}
|
|
}
|
|
|
|
if err := flush(); err != nil {
|
|
return xerrors.Errorf("flush chat stream output: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func TestConsumeChatStreamText(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
events := make(chan codersdk.ChatStreamEvent, 7)
|
|
for _, event := range []codersdk.ChatStreamEvent{
|
|
{Type: codersdk.ChatStreamEventTypeMessagePart, MessagePart: &codersdk.ChatStreamMessagePart{Role: codersdk.ChatMessageRoleAssistant, Part: codersdk.ChatMessagePart{Type: codersdk.ChatMessagePartTypeText, Text: "Hello"}}},
|
|
{Type: codersdk.ChatStreamEventTypeMessagePart, MessagePart: &codersdk.ChatStreamMessagePart{Role: codersdk.ChatMessageRoleAssistant, Part: codersdk.ChatMessagePart{Type: codersdk.ChatMessagePartTypeToolCall, Text: "ignored"}}},
|
|
{Type: codersdk.ChatStreamEventTypeMessagePart, MessagePart: &codersdk.ChatStreamMessagePart{Role: codersdk.ChatMessageRoleAssistant, Part: codersdk.ChatMessagePart{Type: codersdk.ChatMessagePartTypeText, Text: " world"}}},
|
|
{Type: codersdk.ChatStreamEventTypeMessage, Message: &codersdk.ChatMessage{ID: 1, ChatID: uuid.New(), Role: codersdk.ChatMessageRoleAssistant, Content: []codersdk.ChatMessagePart{{Type: codersdk.ChatMessagePartTypeText, Text: "Hello world"}}}},
|
|
{Type: codersdk.ChatStreamEventTypeStatus, Status: &codersdk.ChatStreamStatus{Status: codersdk.ChatStatusRunning}},
|
|
{Type: codersdk.ChatStreamEventTypeRetry, Retry: &codersdk.ChatStreamRetry{Attempt: 2, Error: "rate limited"}},
|
|
{Type: codersdk.ChatStreamEventTypeError, Error: &codersdk.ChatError{Message: "boom"}},
|
|
} {
|
|
events <- event
|
|
}
|
|
close(events)
|
|
|
|
var stdout bytes.Buffer
|
|
var stderr bytes.Buffer
|
|
err := consumeChatStream(events, chatWatchWriters{stdout: &stdout, stderr: &stderr})
|
|
require.NoError(t, err)
|
|
require.Equal(t, "Hello world\n[Status: running]\n[Retry attempt 2 after error: rate limited]\n", stdout.String())
|
|
require.Equal(t, "[Error: boom]\n", stderr.String())
|
|
}
|