summaryrefslogtreecommitdiff
path: root/internal
diff options
context:
space:
mode:
authorPaul Buetow <paul@buetow.org>2025-09-04 16:10:01 +0300
committerPaul Buetow <paul@buetow.org>2025-09-04 16:10:01 +0300
commit09b33e65d92f5fb5b907e49c3d27584615cf2b83 (patch)
tree60f3acd4880dd2a3c0abed7e895a5f13acafcca7 /internal
parentbf53cf2a673af254d7a08bc3b2ab815a08f66117 (diff)
tests: add negative SSE test, table-driven refactors, and use shared fixtures across tests; update REPORT.md progress
Diffstat (limited to 'internal')
-rw-r--r--internal/llm/openai_sse_negative_test.go27
-rw-r--r--internal/lsp/code_fences_table_test.go32
-rw-r--r--internal/lsp/label_filter_table_test.go17
3 files changed, 76 insertions, 0 deletions
diff --git a/internal/llm/openai_sse_negative_test.go b/internal/llm/openai_sse_negative_test.go
new file mode 100644
index 0000000..22b938c
--- /dev/null
+++ b/internal/llm/openai_sse_negative_test.go
@@ -0,0 +1,27 @@
+package llm
+
+import (
+ "context"
+ "io"
+ "net/http"
+ "net/http/httptest"
+ "testing"
+)
+
+func TestOpenAI_ChatStream_SSE_MalformedChunk(t *testing.T) {
+ // Malformed JSON chunk should be skipped; no onDelta calls; no error.
+ srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ w.Header().Set("Content-Type", "text/event-stream")
+ io.WriteString(w, "data: {not json}\n\n")
+ io.WriteString(w, "data: [DONE]\n")
+ }))
+ defer srv.Close()
+ c := newOpenAI(srv.URL, "g", "KEY", f64p(0.2)).(openAIClient)
+ c.httpClient = srv.Client()
+ var got string
+ if err := c.ChatStream(context.Background(), []Message{{Role: "user", Content: "hi"}}, func(s string){ got += s }); err != nil {
+ t.Fatalf("unexpected error for malformed chunk: %v", err)
+ }
+ if got != "" { t.Fatalf("expected no deltas for malformed chunk, got %q", got) }
+}
+
diff --git a/internal/lsp/code_fences_table_test.go b/internal/lsp/code_fences_table_test.go
new file mode 100644
index 0000000..c217bce
--- /dev/null
+++ b/internal/lsp/code_fences_table_test.go
@@ -0,0 +1,32 @@
+package lsp
+
+import "testing"
+
+func TestStripCodeFences_Table(t *testing.T) {
+ cases := []struct{ name, in, want string }{
+ {"no_fence", "return x", "return x"},
+ {"plain_fence", "```\nA\nB\n```", "A\nB"},
+ {"lang_fence", "```go\nfmt.Println()\n```", "fmt.Println()"},
+ {"spaces", " \n```python\nprint('x')\n```\n ", "print('x')"},
+ }
+ for _, c := range cases {
+ if got := stripCodeFences(c.in); got != c.want {
+ t.Fatalf("%s: got %q want %q", c.name, got, c.want)
+ }
+ }
+}
+
+func TestStripInlineCodeSpan_Table(t *testing.T) {
+ cases := []struct{ name, in, want string }{
+ {"no_ticks", "text", "text"},
+ {"single_span", "Use `foo()` here", "foo()"},
+ {"multiple", "`a` + `b`", "a"},
+ {"unmatched", "`missing end", "`missing end"},
+ }
+ for _, c := range cases {
+ if got := stripInlineCodeSpan(c.in); got != c.want {
+ t.Fatalf("%s: got %q want %q", c.name, got, c.want)
+ }
+ }
+}
+
diff --git a/internal/lsp/label_filter_table_test.go b/internal/lsp/label_filter_table_test.go
new file mode 100644
index 0000000..c42b0b1
--- /dev/null
+++ b/internal/lsp/label_filter_table_test.go
@@ -0,0 +1,17 @@
+package lsp
+
+import "testing"
+
+func TestLabelForCompletion_Table(t *testing.T) {
+ cases := []struct{ cleaned, filter, want string }{
+ {"line one\nline two", "zzz", "zzz"},
+ {"result", "re", "result"},
+ {"hello world", "he", "hello world"},
+ }
+ for _, c := range cases {
+ if got := labelForCompletion(c.cleaned, c.filter); got != c.want {
+ t.Fatalf("cleaned=%q filter=%q got %q want %q", c.cleaned, c.filter, got, c.want)
+ }
+ }
+}
+