package main import ( "encoding/json" "io" "net/http" "net/http/httptest" "os" "path/filepath" "strings" "testing" "time" ) func TestParseClaudeJSON(t *testing.T) { t.Parallel() payload, err := parseClaudeJSON("log line\n{\"result\":\"review ok\",\"is_error\":false}\n") if err != nil { t.Fatalf("parseClaudeJSON returned error: %v", err) } if got := payload["result"]; got != "review ok" { t.Fatalf("unexpected result: %v", got) } } func TestCallOpenAICompatible(t *testing.T) { t.Parallel() server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { if got := r.Header.Get("Authorization"); got != "Bearer test-key" { t.Fatalf("unexpected auth header: %s", got) } var body map[string]any if err := json.NewDecoder(r.Body).Decode(&body); err != nil { t.Fatalf("decode request body: %v", err) } if got := body["model"]; got != "qwen2.5-coder:latest" { t.Fatalf("unexpected model: %v", got) } _ = json.NewEncoder(w).Encode(map[string]any{ "choices": []map[string]any{ { "message": map[string]any{ "content": "review ok", }, }, }, }) })) defer server.Close() output, err := callOpenAICompatible( server.URL, "test-key", "qwen2.5-coder:latest", []map[string]string{ {"role": "user", "content": "hello"}, }, ) if err != nil { t.Fatalf("callOpenAICompatible returned error: %v", err) } if output != "review ok" { t.Fatalf("unexpected output: %s", output) } } func TestHandleChatToolRequiresPrompt(t *testing.T) { t.Setenv("LLM_API_KEY", "test-key") t.Setenv("LLM_BASE_URL", "http://127.0.0.1:11434/v1") _, err := handleChatTool(map[string]any{}) if err == nil || err.Error() != "prompt is required" { t.Fatalf("expected prompt error, got %v", err) } } func TestParseClaudeJSONReturnsErrorForPlainText(t *testing.T) { t.Parallel() _, err := parseClaudeJSON("plain text only\n") if err == nil { t.Fatal("expected parse error for plain text output") } } func TestCallOpenAICompatibleReturnsStatusError(t *testing.T) { t.Parallel() server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { http.Error(w, "bad gateway", http.StatusBadGateway) })) defer server.Close() _, err := callOpenAICompatible( server.URL, "test-key", "qwen2.5-coder:latest", []map[string]string{{"role": "user", "content": "hello"}}, ) if err == nil || err.Error() == "" { t.Fatal("expected non-2xx status error") } } func TestRunClaudeReviewSurfacesCliExitFailure(t *testing.T) { tempDir := t.TempDir() cliPath := filepath.Join(tempDir, "claude") if err := os.WriteFile(cliPath, []byte("#!/bin/sh\necho boom >&2\nexit 2\n"), 0o755); err != nil { t.Fatalf("write fake claude script: %v", err) } t.Setenv("CLAUDE_BIN", cliPath) _, err := runClaudeReview("review this", "", "", "", 2*time.Second) if err == nil || err.Error() == "" { t.Fatal("expected cli failure") } } func TestRunClaudeReviewSurfacesNonJSONStdout(t *testing.T) { tempDir := t.TempDir() cliPath := filepath.Join(tempDir, "claude") if err := os.WriteFile(cliPath, []byte("#!/bin/sh\necho plain-text-output\nexit 0\n"), 0o755); err != nil { t.Fatalf("write fake claude script: %v", err) } t.Setenv("CLAUDE_BIN", cliPath) _, err := runClaudeReview("review this", "", "", "", 2*time.Second) if err == nil || err.Error() == "" { t.Fatal("expected non-json stdout error") } } func TestPrintBridgeVersionInfoMatchesPingContract(t *testing.T) { buildCommit = "cfd19f6" buildVersion = "v1.0-beta2" buildDate = "2026-04-22T12:23:28+08:00" t.Cleanup(func() { buildCommit = "" buildVersion = "v1.0-beta2" buildDate = "" }) output := captureStdout(t, printBridgeVersionInfo) var payload map[string]any if err := json.Unmarshal([]byte(output), &payload); err != nil { t.Fatalf("decode version payload: %v", err) } if payload["status"] != "ok" { t.Fatalf("expected status ok, got %#v", payload["status"]) } if payload["commit"] != "cfd19f6" { t.Fatalf("unexpected commit: %#v", payload["commit"]) } if payload["version"] != "v1.0-beta2" { t.Fatalf("unexpected version: %#v", payload["version"]) } if payload["build-date"] != "2026-04-22T12:23:28+08:00" { t.Fatalf("unexpected build-date: %#v", payload["build-date"]) } } func TestPrintBridgeVersionInfoUsesBuildVariables(t *testing.T) { buildCommit = "deadbee" buildVersion = "v1.0-beta2" buildDate = "2026-04-01T01:02:03+08:00" t.Cleanup(func() { buildCommit = "" buildVersion = "v1.0-beta2" buildDate = "" }) output := captureStdout(t, printBridgeVersionInfo) if !strings.Contains(output, `"commit":"deadbee"`) { t.Fatalf("unexpected output: %q", output) } if !strings.Contains(output, `"version":"v1.0-beta2"`) { t.Fatalf("unexpected output: %q", output) } if !strings.Contains(output, `"build-date":"2026-04-01T01:02:03+08:00"`) { t.Fatalf("unexpected output: %q", output) } } func captureStdout(t *testing.T, fn func() error) string { t.Helper() old := os.Stdout r, w, err := os.Pipe() if err != nil { t.Fatalf("pipe stdout: %v", err) } os.Stdout = w defer func() { os.Stdout = old }() done := make(chan string, 1) go func() { var builder strings.Builder _, _ = io.Copy(&builder, r) done <- builder.String() }() callErr := fn() _ = w.Close() out := <-done _ = r.Close() if callErr != nil { t.Fatalf("call failed: %v", callErr) } return strings.TrimSpace(out) }