140 lines
3.2 KiB
Go

package handlers
import (
"config-lsp/common"
docvalues "config-lsp/doc-values"
sshconfig "config-lsp/handlers/ssh_config"
"config-lsp/handlers/ssh_config/ast"
"config-lsp/handlers/ssh_config/fields"
"config-lsp/utils"
protocol "github.com/tliron/glsp/protocol_3_16"
)
func GetRootCompletions(
d *sshconfig.SSHDocument,
parentBlock ast.SSHBlock,
suggestValue bool,
) []protocol.CompletionItem {
kind := protocol.CompletionItemKindField
availableOptions := make(map[fields.NormalizedOptionName]docvalues.DocumentationValue, 0)
for key, option := range fields.Options {
// Check for duplicates
if d.DoesOptionExist(key, parentBlock) && !utils.KeyExists(fields.AllowedDuplicateOptions, key) {
continue
}
if parentBlock != nil && parentBlock.GetBlockType() == ast.SSHBlockTypeHost && utils.KeyExists(fields.HostDisallowedOptions, key) {
continue
}
if parentBlock == nil && utils.KeyExists(fields.GlobalDisallowedOptions, key) {
continue
}
availableOptions[key] = option
}
return utils.MapMapToSlice(
availableOptions,
func(normalizedName fields.NormalizedOptionName, doc docvalues.DocumentationValue) protocol.CompletionItem {
name := fields.FieldsNameFormattedMap[normalizedName]
completion := &protocol.CompletionItem{
Label: string(name),
Kind: &kind,
Documentation: doc.Documentation,
}
if suggestValue {
format := protocol.InsertTextFormatSnippet
insertText := string(name) + " " + "${1:value}"
completion.InsertTextFormat = &format
completion.InsertText = &insertText
}
return *completion
},
)
}
func GetOptionCompletions(
d *sshconfig.SSHDocument,
entry *ast.SSHOption,
block ast.SSHBlock,
line uint32,
cursor common.CursorPosition,
) []protocol.CompletionItem {
option, found := fields.Options[entry.Key.Key]
if !found {
return nil
}
if entry.Key.Key == matchOption {
matchBlock := block.(*ast.SSHMatchBlock)
return getMatchCompletions(
d,
cursor,
matchBlock.MatchValue,
)
}
if entry.Key.Key == tagOption {
return getTagCompletions(
d,
line,
cursor,
entry,
)
}
if entry.OptionValue == nil {
return option.DeprecatedFetchCompletions("", 0)
}
// token completions
completions := getTokenCompletions(entry, cursor)
// Hello wo|rld
lineValue := entry.OptionValue.Value.Raw
// NEW: docvalues index
completions = append(completions, option.DeprecatedFetchCompletions(
lineValue,
common.DeprecatedImprovedCursorToIndex(
cursor,
lineValue,
entry.OptionValue.Start.Character,
),
)...)
return completions
}
func getTokenCompletions(
entry *ast.SSHOption,
cursor common.CursorPosition,
) []protocol.CompletionItem {
completions := make([]protocol.CompletionItem, 0)
index := common.CursorToCharacterIndex(uint32(cursor))
if entry.Value.Raw[index] == '%' {
if tokens, found := fields.OptionsTokensMap[entry.Key.Key]; found {
for _, token := range tokens {
description := fields.AvailableTokens[token]
kind := protocol.CompletionItemKindConstant
completions = append(completions, protocol.CompletionItem{
Label: token,
Kind: &kind,
Documentation: description,
})
}
}
}
return completions
}