Skip to content

Commit 68fb648

Browse files
committed
terraform/lang: Implement resource completion
Closes #9
1 parent 30da977 commit 68fb648

8 files changed

+612
-440
lines changed

internal/terraform/lang/config_block.go

+110
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,12 @@
11
package lang
22

33
import (
4+
"fmt"
5+
"log"
6+
47
hcl "github.com/hashicorp/hcl/v2"
58
"github.com/hashicorp/hcl/v2/hclsyntax"
9+
tfjson "github.com/hashicorp/terraform-json"
610
lsp "github.com/sourcegraph/go-lsp"
711
)
812

@@ -15,3 +19,109 @@ type ConfigBlock interface {
1519
type configBlockFactory interface {
1620
New(*hclsyntax.Block) (ConfigBlock, error)
1721
}
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

Comments
 (0)