1
1
package lang
2
2
3
3
import (
4
+ "fmt"
5
+ "log"
6
+
4
7
hcl "github.com/hashicorp/hcl/v2"
5
8
"github.com/hashicorp/hcl/v2/hclsyntax"
9
+ tfjson "github.com/hashicorp/terraform-json"
6
10
lsp "github.com/sourcegraph/go-lsp"
7
11
)
8
12
@@ -15,3 +19,109 @@ type ConfigBlock interface {
15
19
type configBlockFactory interface {
16
20
New (* hclsyntax.Block ) (ConfigBlock , error )
17
21
}
22
+
23
+ type completableBlock struct {
24
+ logger * log.Logger
25
+ caps lsp.TextDocumentClientCapabilities
26
+ hclBlock * hclsyntax.Block
27
+ schema * tfjson.SchemaBlock
28
+ }
29
+
30
+ func (cb * completableBlock ) completionItemsAtPos (pos hcl.Pos ) (lsp.CompletionList , error ) {
31
+ list := lsp.CompletionList {}
32
+
33
+ cb .logger .Printf ("block: %#v" , cb .hclBlock )
34
+
35
+ block := ParseBlock (cb .hclBlock , cb .schema )
36
+
37
+ if ! block .PosInBody (pos ) {
38
+ // Avoid autocompleting outside of body, for now
39
+ cb .logger .Println ("avoiding completion outside of block body" )
40
+ return list , nil
41
+ }
42
+
43
+ if block .PosInAttribute (pos ) {
44
+ cb .logger .Println ("avoiding completion in the middle of existing attribute" )
45
+ return list , nil
46
+ }
47
+
48
+ b , ok := block .BlockAtPos (pos )
49
+ if ! ok {
50
+ // This should never happen as the completion
51
+ // should only be called on a block the "pos" points to
52
+ cb .logger .Printf ("block type not found at %#v" , pos )
53
+ return list , nil
54
+ }
55
+
56
+ for name , attr := range b .Attributes () {
57
+ if attr .IsComputedOnly () || attr .IsDeclared () {
58
+ continue
59
+ }
60
+ list .Items = append (list .Items , cb .completionItemForAttr (name , attr , pos ))
61
+ }
62
+
63
+ for name , block := range b .BlockTypes () {
64
+ if block .ReachedMaxItems () {
65
+ continue
66
+ }
67
+ list .Items = append (list .Items , cb .completionItemForNestedBlock (name , block , pos ))
68
+ }
69
+
70
+ sortCompletionItems (list .Items )
71
+
72
+ return list , nil
73
+ }
74
+
75
+ func (cb * completableBlock ) completionItemForAttr (name string , attr * Attribute , pos hcl.Pos ) lsp.CompletionItem {
76
+ snippetSupport := cb .caps .Completion .CompletionItem .SnippetSupport
77
+
78
+ if snippetSupport {
79
+ return lsp.CompletionItem {
80
+ Label : name ,
81
+ Kind : lsp .CIKField ,
82
+ InsertTextFormat : lsp .ITFSnippet ,
83
+ Detail : schemaAttributeDetail (attr .Schema ()),
84
+ TextEdit : & lsp.TextEdit {
85
+ Range : lsp.Range {
86
+ Start : lsp.Position {Line : pos .Line - 1 , Character : pos .Column - 1 },
87
+ End : lsp.Position {Line : pos .Line - 1 , Character : pos .Column - 1 },
88
+ },
89
+ NewText : fmt .Sprintf ("%s = %s" , name , snippetForAttrType (0 , attr .Schema ().AttributeType )),
90
+ },
91
+ }
92
+ }
93
+
94
+ return lsp.CompletionItem {
95
+ Label : name ,
96
+ Kind : lsp .CIKField ,
97
+ InsertTextFormat : lsp .ITFPlainText ,
98
+ Detail : schemaAttributeDetail (attr .Schema ()),
99
+ }
100
+ }
101
+
102
+ func (cb * completableBlock ) completionItemForNestedBlock (name string , blockType * BlockType , pos hcl.Pos ) lsp.CompletionItem {
103
+ snippetSupport := cb .caps .Completion .CompletionItem .SnippetSupport
104
+
105
+ if snippetSupport {
106
+ return lsp.CompletionItem {
107
+ Label : name ,
108
+ Kind : lsp .CIKField ,
109
+ InsertTextFormat : lsp .ITFSnippet ,
110
+ Detail : schemaBlockDetail (blockType ),
111
+ TextEdit : & lsp.TextEdit {
112
+ Range : lsp.Range {
113
+ Start : lsp.Position {Line : pos .Line - 1 , Character : pos .Column - 1 },
114
+ End : lsp.Position {Line : pos .Line - 1 , Character : pos .Column - 1 },
115
+ },
116
+ NewText : snippetForNestedBlock (name ),
117
+ },
118
+ }
119
+ }
120
+
121
+ return lsp.CompletionItem {
122
+ Label : name ,
123
+ Kind : lsp .CIKField ,
124
+ InsertTextFormat : lsp .ITFPlainText ,
125
+ Detail : schemaBlockDetail (blockType ),
126
+ }
127
+ }
0 commit comments