summaryrefslogtreecommitdiff
path: root/internal/lsp/provider_native_success_test.go
blob: ac227becc3058ccb2501462a9e72b597041355f6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
package lsp

import (
	"context"
	"testing"

	"codeberg.org/snonux/hexai/internal/llm"
)

type fakeCompleterOk struct{}

func (fakeCompleterOk) Chat(context.Context, []llm.Message, ...llm.RequestOption) (string, error) {
	return "", nil
}
func (fakeCompleterOk) Name() string         { return "prov" }
func (fakeCompleterOk) DefaultModel() string { return "m" }
func (fakeCompleterOk) CodeCompletion(context.Context, string, string, int, string, float64) ([]string, error) {
	return []string{"SUGG"}, nil
}

func TestProviderNativeCompletion_Success(t *testing.T) {
	s := newTestServer()
	s.llmClient = fakeCompleterOk{}
	// current line with dot trigger; position after dot
	current := "fmt."
	p := CompletionParams{TextDocument: TextDocumentIdentifier{URI: "file:///x.go"}, Position: Position{Line: 0, Character: len(current)}}
	items, ok := s.tryProviderNativeCompletion(current, p, "", "", "func f(){}", "doc", false, "", false)
	if !ok || len(items) == 0 {
		t.Fatalf("expected provider-native items")
	}
	if items[0].Label == "" || items[0].TextEdit == nil {
		t.Fatalf("unexpected completion item: %+v", items[0])
	}
}

type fakeCompleterIndent struct{}

func (fakeCompleterIndent) Chat(context.Context, []llm.Message, ...llm.RequestOption) (string, error) {
	return "", nil
}
func (fakeCompleterIndent) Name() string         { return "prov" }
func (fakeCompleterIndent) DefaultModel() string { return "m" }
func (fakeCompleterIndent) CodeCompletion(context.Context, string, string, int, string, float64) ([]string, error) {
	return []string{"a\nb"}, nil
}

func TestProviderNativeCompletion_IndentWithDoubleOpen(t *testing.T) {
	s := newTestServer()
	s.llmClient = fakeCompleterIndent{}
	current := "  >>do>" // leading indent + double-open marker
	p := CompletionParams{TextDocument: TextDocumentIdentifier{URI: "file:///x.go"}, Position: Position{Line: 0, Character: len(current)}}
	items, ok := s.tryProviderNativeCompletion(current, p, "", "", "func f(){}", "doc", false, "", false)
	if !ok || len(items) == 0 {
		t.Fatalf("expected provider-native items")
	}
	if items[0].TextEdit == nil {
		t.Fatalf("expected text edit")
	}
	if got := items[0].TextEdit.NewText; len(got) < 2 || got[:2] != "  " {
		t.Fatalf("expected indentation applied, got %q", got)
	}
}

type fakeCompleterCapture struct{ lastPrompt string }

func (fakeCompleterCapture) Chat(context.Context, []llm.Message, ...llm.RequestOption) (string, error) {
	return "", nil
}
func (fakeCompleterCapture) Name() string         { return "prov" }
func (fakeCompleterCapture) DefaultModel() string { return "m" }
func (f *fakeCompleterCapture) CodeCompletion(_ context.Context, prompt string, suffix string, n int, language string, temperature float64) ([]string, error) {
	f.lastPrompt = prompt
	return []string{"SUG"}, nil
}

func TestProviderNativeCompletion_UsesPromptTemplate(t *testing.T) {
	s := newTestServer()
	cap := &fakeCompleterCapture{}
	s.llmClient = cap
	s.promptNativeCompletion = "NATIVE {{path}} {{before}}"
	uri := "file:///x.go"
	s.setDocument(uri, "AAA\nBBB\nCCC")
	current := "fmt."
	// Cursor at line 1, char 1 -> before should be "AAA\nB"
	p := CompletionParams{TextDocument: TextDocumentIdentifier{URI: uri}, Position: Position{Line: 1, Character: 1}}
	if _, ok := s.tryProviderNativeCompletion(current, p, "", "", "func f(){}", "doc", false, "", false); !ok {
		t.Fatalf("expected provider-native path")
	}
	if cap.lastPrompt == "" {
		t.Fatalf("expected captured prompt")
	}
	if cap.lastPrompt != "NATIVE /x.go AAA\nB" {
		t.Fatalf("unexpected prompt: %q", cap.lastPrompt)
	}
}