package claude import ( "ds2api/internal/sse" "encoding/json" "io" "net/http" "net/http/httptest" "strings" "testing" "time" ) type claudeFrame struct { Event string Payload map[string]any } func makeClaudeSSEHTTPResponse(lines ...string) *http.Response { body := strings.Join(lines, "\n") if !strings.HasSuffix(body, "\n") { body += "\n" } return &http.Response{ StatusCode: http.StatusOK, Header: make(http.Header), Body: io.NopCloser(strings.NewReader(body)), } } func parseClaudeFrames(t *testing.T, body string) []claudeFrame { t.Helper() chunks := strings.Split(body, "\n\n") frames := make([]claudeFrame, 0, len(chunks)) for _, chunk := range chunks { chunk = strings.TrimSpace(chunk) if chunk == "" { continue } lines := strings.Split(chunk, "\n") eventName := "" dataPayload := "" for _, line := range lines { line = strings.TrimSpace(line) switch { case strings.HasPrefix(line, "event:"): eventName = strings.TrimSpace(strings.TrimPrefix(line, "event:")) case strings.HasPrefix(line, "data:"): dataPayload = strings.TrimSpace(strings.TrimPrefix(line, "data:")) } } if eventName == "" || dataPayload == "" { continue } var payload map[string]any if err := json.Unmarshal([]byte(dataPayload), &payload); err != nil { t.Fatalf("decode frame failed: %v, payload=%s", err, dataPayload) } frames = append(frames, claudeFrame{Event: eventName, Payload: payload}) } return frames } func findClaudeFrames(frames []claudeFrame, event string) []claudeFrame { out := make([]claudeFrame, 0) for _, f := range frames { if f.Event == event { out = append(out, f) } } return out } func TestHandleClaudeStreamRealtimeTextIncrementsWithEventHeaders(t *testing.T) { h := &Handler{} resp := makeClaudeSSEHTTPResponse( `data: {"p":"response/content","v":"Hel"}`, `data: {"p":"response/content","v":"lo"}`, `data: [DONE]`, ) rec := httptest.NewRecorder() req := httptest.NewRequest(http.MethodPost, "/anthropic/v1/messages", nil) h.handleClaudeStreamRealtime(rec, req, resp, "claude-sonnet-4-5", []any{map[string]any{"role": "user", "content": "hi"}}, false, false, nil) body := rec.Body.String() if !strings.Contains(body, "event: message_start") { t.Fatalf("missing event header: message_start, body=%s", body) } if !strings.Contains(body, "event: content_block_delta") { t.Fatalf("missing event header: content_block_delta, body=%s", body) } if !strings.Contains(body, "event: message_stop") { t.Fatalf("missing event header: message_stop, body=%s", body) } frames := parseClaudeFrames(t, body) deltas := findClaudeFrames(frames, "content_block_delta") if len(deltas) < 2 { t.Fatalf("expected at least 2 text deltas, got=%d body=%s", len(deltas), body) } combined := strings.Builder{} for _, f := range deltas { delta, _ := f.Payload["delta"].(map[string]any) if delta["type"] == "text_delta" { combined.WriteString(asString(delta["text"])) } } if combined.String() != "Hello" { t.Fatalf("unexpected combined text: %q body=%s", combined.String(), body) } } func TestHandleClaudeStreamRealtimeThinkingDelta(t *testing.T) { h := &Handler{} resp := makeClaudeSSEHTTPResponse( `data: {"p":"response/thinking_content","v":"思"}`, `data: {"p":"response/thinking_content","v":"考"}`, `data: {"p":"response/content","v":"ok"}`, `data: [DONE]`, ) rec := httptest.NewRecorder() req := httptest.NewRequest(http.MethodPost, "/anthropic/v1/messages", nil) h.handleClaudeStreamRealtime(rec, req, resp, "claude-sonnet-4-5", []any{map[string]any{"role": "user", "content": "hi"}}, true, false, nil) frames := parseClaudeFrames(t, rec.Body.String()) foundThinkingDelta := false for _, f := range findClaudeFrames(frames, "content_block_delta") { delta, _ := f.Payload["delta"].(map[string]any) if delta["type"] == "thinking_delta" { foundThinkingDelta = true break } } if !foundThinkingDelta { t.Fatalf("expected thinking_delta event, body=%s", rec.Body.String()) } } func TestHandleClaudeStreamRealtimeToolSafety(t *testing.T) { h := &Handler{} resp := makeClaudeSSEHTTPResponse( `data: {"p":"response/content","v":"{\"tool_calls\":[{\"name\":\"search\""}`, `data: {"p":"response/content","v":",\"input\":{\"q\":\"go\"}}]}"}`, `data: [DONE]`, ) rec := httptest.NewRecorder() req := httptest.NewRequest(http.MethodPost, "/anthropic/v1/messages", nil) h.handleClaudeStreamRealtime(rec, req, resp, "claude-sonnet-4-5", []any{map[string]any{"role": "user", "content": "use tool"}}, false, false, []string{"search"}) frames := parseClaudeFrames(t, rec.Body.String()) for _, f := range findClaudeFrames(frames, "content_block_delta") { delta, _ := f.Payload["delta"].(map[string]any) if delta["type"] == "text_delta" && strings.Contains(asString(delta["text"]), `"tool_calls"`) { t.Fatalf("raw tool_calls JSON leaked in text delta: body=%s", rec.Body.String()) } } foundToolUse := false for _, f := range findClaudeFrames(frames, "content_block_start") { contentBlock, _ := f.Payload["content_block"].(map[string]any) if contentBlock["type"] == "tool_use" { foundToolUse = true break } } if !foundToolUse { t.Fatalf("expected tool_use block in stream, body=%s", rec.Body.String()) } foundToolUseStop := false for _, f := range findClaudeFrames(frames, "message_delta") { delta, _ := f.Payload["delta"].(map[string]any) if delta["stop_reason"] == "tool_use" { foundToolUseStop = true break } } if !foundToolUseStop { t.Fatalf("expected stop_reason=tool_use, body=%s", rec.Body.String()) } } func TestHandleClaudeStreamRealtimeUpstreamErrorEvent(t *testing.T) { h := &Handler{} resp := makeClaudeSSEHTTPResponse( `data: {"error":{"message":"boom"}}`, ) rec := httptest.NewRecorder() req := httptest.NewRequest(http.MethodPost, "/anthropic/v1/messages", nil) h.handleClaudeStreamRealtime(rec, req, resp, "claude-sonnet-4-5", []any{map[string]any{"role": "user", "content": "hi"}}, false, false, nil) frames := parseClaudeFrames(t, rec.Body.String()) errFrames := findClaudeFrames(frames, "error") if len(errFrames) == 0 { t.Fatalf("expected error event frame, body=%s", rec.Body.String()) } if errFrames[0].Payload["type"] != "error" { t.Fatalf("expected error payload type, body=%s", rec.Body.String()) } } func TestHandleClaudeStreamRealtimePingEvent(t *testing.T) { h := &Handler{} oldPing := claudeStreamPingInterval oldIdle := claudeStreamIdleTimeout oldKeepalive := claudeStreamMaxKeepaliveCnt claudeStreamPingInterval = 10 * time.Millisecond claudeStreamIdleTimeout = 300 * time.Millisecond claudeStreamMaxKeepaliveCnt = 50 defer func() { claudeStreamPingInterval = oldPing claudeStreamIdleTimeout = oldIdle claudeStreamMaxKeepaliveCnt = oldKeepalive }() pr, pw := io.Pipe() resp := &http.Response{StatusCode: http.StatusOK, Header: make(http.Header), Body: pr} go func() { time.Sleep(40 * time.Millisecond) _, _ = io.WriteString(pw, "data: {\"p\":\"response/content\",\"v\":\"hi\"}\n") _, _ = io.WriteString(pw, "data: [DONE]\n") _ = pw.Close() }() rec := httptest.NewRecorder() req := httptest.NewRequest(http.MethodPost, "/anthropic/v1/messages", nil) h.handleClaudeStreamRealtime(rec, req, resp, "claude-sonnet-4-5", []any{map[string]any{"role": "user", "content": "hi"}}, false, false, nil) frames := parseClaudeFrames(t, rec.Body.String()) if len(findClaudeFrames(frames, "ping")) == 0 { t.Fatalf("expected ping event in stream, body=%s", rec.Body.String()) } } func TestCollectDeepSeekRegression(t *testing.T) { resp := makeClaudeSSEHTTPResponse( `data: {"p":"response/thinking_content","v":"想"}`, `data: {"p":"response/content","v":"答"}`, `data: [DONE]`, ) result := sse.CollectStream(resp, true, true) if result.Thinking != "想" { t.Fatalf("unexpected thinking: %q", result.Thinking) } if result.Text != "答" { t.Fatalf("unexpected text: %q", result.Text) } } func asString(v any) string { s, _ := v.(string) return s }