2022-11-04 03:16:14 +01:00
|
|
|
// scripthaus completion
|
|
|
|
package comp
|
|
|
|
|
|
|
|
import (
|
2022-11-10 22:52:51 +01:00
|
|
|
"bytes"
|
2022-11-04 03:16:14 +01:00
|
|
|
"context"
|
|
|
|
"fmt"
|
2022-11-10 22:52:51 +01:00
|
|
|
"sort"
|
|
|
|
"strconv"
|
2022-11-04 03:16:14 +01:00
|
|
|
"strings"
|
|
|
|
"unicode"
|
2022-11-22 09:26:41 +01:00
|
|
|
"unicode/utf8"
|
2022-11-04 03:16:14 +01:00
|
|
|
|
2023-07-26 23:24:14 +02:00
|
|
|
"github.com/commandlinedev/apishell/pkg/simpleexpand"
|
|
|
|
"github.com/commandlinedev/prompt-server/pkg/shparse"
|
|
|
|
"github.com/commandlinedev/prompt-server/pkg/sstore"
|
|
|
|
"github.com/commandlinedev/prompt-server/pkg/utilfn"
|
2022-11-04 03:16:14 +01:00
|
|
|
"mvdan.cc/sh/v3/syntax"
|
|
|
|
)
|
|
|
|
|
2022-11-10 22:52:51 +01:00
|
|
|
const MaxCompQuoteLen = 5000
|
|
|
|
|
2022-11-04 03:16:14 +01:00
|
|
|
const (
|
2022-11-11 01:00:51 +01:00
|
|
|
// local to simplecomp
|
|
|
|
CGTypeCommand = "command"
|
|
|
|
CGTypeFile = "file"
|
|
|
|
CGTypeDir = "directory"
|
|
|
|
CGTypeVariable = "variable"
|
|
|
|
|
|
|
|
// implemented in cmdrunner
|
|
|
|
CGTypeMeta = "metacmd"
|
|
|
|
CGTypeCommandMeta = "command+meta"
|
|
|
|
|
|
|
|
CGTypeRemote = "remote"
|
|
|
|
CGTypeRemoteInstance = "remoteinstance"
|
|
|
|
CGTypeGlobalCmd = "globalcmd"
|
2022-11-04 03:16:14 +01:00
|
|
|
)
|
|
|
|
|
2022-11-10 22:52:51 +01:00
|
|
|
const (
|
|
|
|
QuoteTypeLiteral = ""
|
|
|
|
QuoteTypeDQ = "\""
|
|
|
|
QuoteTypeANSI = "$'"
|
|
|
|
QuoteTypeSQ = "'"
|
|
|
|
)
|
|
|
|
|
2022-11-10 05:38:28 +01:00
|
|
|
type CompContext struct {
|
2022-11-11 03:51:20 +01:00
|
|
|
RemotePtr *sstore.RemotePtrType
|
2022-11-29 03:03:02 +01:00
|
|
|
Cwd string
|
2022-11-10 05:38:28 +01:00
|
|
|
ForDisplay bool
|
|
|
|
}
|
|
|
|
|
2022-11-04 03:16:14 +01:00
|
|
|
type ParsedWord struct {
|
2022-11-10 05:38:28 +01:00
|
|
|
Offset int
|
|
|
|
Word *syntax.Word
|
|
|
|
PartialWord string
|
|
|
|
Prefix string
|
2022-11-04 03:16:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type CompPoint struct {
|
2022-11-10 05:38:28 +01:00
|
|
|
StmtStr string
|
2022-11-04 03:16:14 +01:00
|
|
|
Words []ParsedWord
|
|
|
|
CompWord int
|
|
|
|
CompWordPos int
|
|
|
|
Prefix string
|
|
|
|
Suffix string
|
|
|
|
}
|
|
|
|
|
2022-11-10 22:52:51 +01:00
|
|
|
// directories will have a trailing "/"
|
|
|
|
type CompEntry struct {
|
|
|
|
Word string
|
|
|
|
IsMetaCmd bool
|
|
|
|
}
|
|
|
|
|
|
|
|
type CompReturn struct {
|
2022-11-11 03:51:20 +01:00
|
|
|
CompType string
|
|
|
|
Entries []CompEntry
|
|
|
|
HasMore bool
|
2022-11-10 22:52:51 +01:00
|
|
|
}
|
|
|
|
|
2022-11-22 09:26:41 +01:00
|
|
|
var noEscChars []bool
|
|
|
|
var specialEsc []string
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
noEscChars = make([]bool, 256)
|
|
|
|
for ch := 0; ch < 256; ch++ {
|
|
|
|
if (ch >= '0' && ch <= '9') || (ch >= 'a' && ch <= 'z') || (ch >= 'A' && ch <= 'Z') ||
|
|
|
|
ch == '-' || ch == '.' || ch == '/' || ch == ':' || ch == '=' {
|
|
|
|
noEscChars[byte(ch)] = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
specialEsc = make([]string, 256)
|
|
|
|
specialEsc[0x7] = "\\a"
|
|
|
|
specialEsc[0x8] = "\\b"
|
|
|
|
specialEsc[0x9] = "\\t"
|
|
|
|
specialEsc[0xa] = "\\n"
|
|
|
|
specialEsc[0xb] = "\\v"
|
|
|
|
specialEsc[0xc] = "\\f"
|
|
|
|
specialEsc[0xd] = "\\r"
|
|
|
|
specialEsc[0x1b] = "\\E"
|
|
|
|
}
|
|
|
|
|
2022-11-10 22:52:51 +01:00
|
|
|
func compQuoteDQString(s string, close bool) string {
|
|
|
|
var buf bytes.Buffer
|
|
|
|
buf.WriteByte('"')
|
|
|
|
for _, ch := range s {
|
|
|
|
if ch == '"' || ch == '\\' || ch == '$' || ch == '`' {
|
|
|
|
buf.WriteByte('\\')
|
|
|
|
buf.WriteRune(ch)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
buf.WriteRune(ch)
|
|
|
|
}
|
|
|
|
if close {
|
|
|
|
buf.WriteByte('"')
|
|
|
|
}
|
|
|
|
return buf.String()
|
|
|
|
}
|
|
|
|
|
2022-11-22 09:26:41 +01:00
|
|
|
func hasGlob(s string) bool {
|
|
|
|
var lastExtGlob bool
|
|
|
|
for _, ch := range s {
|
|
|
|
if ch == '*' || ch == '?' || ch == '[' || ch == '{' {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if ch == '+' || ch == '@' || ch == '!' {
|
|
|
|
lastExtGlob = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if lastExtGlob && ch == '(' {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
lastExtGlob = false
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func writeUtf8Literal(buf *bytes.Buffer, ch rune) {
|
|
|
|
var runeArr [utf8.UTFMax]byte
|
|
|
|
buf.WriteString("$'")
|
|
|
|
barr := runeArr[:]
|
|
|
|
byteLen := utf8.EncodeRune(barr, ch)
|
|
|
|
for i := 0; i < byteLen; i++ {
|
|
|
|
buf.WriteString("\\x")
|
|
|
|
buf.WriteByte(utilfn.HexDigits[barr[i]/16])
|
|
|
|
buf.WriteByte(utilfn.HexDigits[barr[i]%16])
|
|
|
|
}
|
|
|
|
buf.WriteByte('\'')
|
|
|
|
}
|
|
|
|
|
|
|
|
func compQuoteLiteralString(s string) string {
|
|
|
|
var buf bytes.Buffer
|
|
|
|
for idx, ch := range s {
|
|
|
|
if ch == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if idx == 0 && ch == '~' {
|
|
|
|
buf.WriteRune(ch)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if ch > unicode.MaxASCII {
|
|
|
|
writeUtf8Literal(&buf, ch)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
var bch = byte(ch)
|
|
|
|
if noEscChars[bch] {
|
|
|
|
buf.WriteRune(ch)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if specialEsc[bch] != "" {
|
|
|
|
buf.WriteString(specialEsc[bch])
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if !unicode.IsPrint(ch) {
|
|
|
|
writeUtf8Literal(&buf, ch)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
buf.WriteByte('\\')
|
|
|
|
buf.WriteByte(bch)
|
|
|
|
}
|
|
|
|
return buf.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func compQuoteSQString(s string) string {
|
|
|
|
var buf bytes.Buffer
|
|
|
|
for _, ch := range s {
|
|
|
|
if ch == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if ch == '\'' {
|
|
|
|
buf.WriteString("'\\''")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
var bch byte
|
|
|
|
if ch <= unicode.MaxASCII {
|
|
|
|
bch = byte(ch)
|
|
|
|
}
|
|
|
|
if ch > unicode.MaxASCII || !unicode.IsPrint(ch) {
|
|
|
|
buf.WriteByte('\'')
|
|
|
|
if bch != 0 && specialEsc[bch] != "" {
|
|
|
|
buf.WriteString(specialEsc[bch])
|
|
|
|
} else {
|
|
|
|
writeUtf8Literal(&buf, ch)
|
|
|
|
}
|
|
|
|
buf.WriteByte('\'')
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
buf.WriteByte(bch)
|
|
|
|
}
|
|
|
|
return buf.String()
|
|
|
|
}
|
|
|
|
|
2022-11-10 22:52:51 +01:00
|
|
|
func compQuoteString(s string, quoteType string, close bool) string {
|
2022-11-22 09:26:41 +01:00
|
|
|
if quoteType != QuoteTypeANSI && quoteType != QuoteTypeLiteral {
|
2022-11-10 22:52:51 +01:00
|
|
|
for _, ch := range s {
|
|
|
|
if ch > unicode.MaxASCII || !unicode.IsPrint(ch) || ch == '!' {
|
|
|
|
quoteType = QuoteTypeANSI
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if ch == '\'' {
|
2022-11-22 09:26:41 +01:00
|
|
|
if quoteType == QuoteTypeSQ {
|
2022-11-10 22:52:51 +01:00
|
|
|
quoteType = QuoteTypeANSI
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if quoteType == QuoteTypeANSI {
|
|
|
|
rtn := strconv.QuoteToASCII(s)
|
|
|
|
rtn = "$'" + strings.ReplaceAll(rtn[1:len(rtn)-1], "'", "\\'")
|
|
|
|
if close {
|
|
|
|
rtn = rtn + "'"
|
|
|
|
}
|
|
|
|
return rtn
|
|
|
|
}
|
|
|
|
if quoteType == QuoteTypeLiteral {
|
2022-11-22 09:26:41 +01:00
|
|
|
return compQuoteLiteralString(s)
|
2022-11-10 22:52:51 +01:00
|
|
|
}
|
|
|
|
if quoteType == QuoteTypeSQ {
|
2022-11-11 00:28:39 +01:00
|
|
|
rtn := utilfn.ShellQuote(s, false, MaxCompQuoteLen)
|
|
|
|
if len(rtn) > 0 && rtn[0] != '\'' {
|
|
|
|
rtn = "'" + rtn + "'"
|
|
|
|
}
|
2022-11-10 22:52:51 +01:00
|
|
|
if !close {
|
|
|
|
rtn = rtn[0 : len(rtn)-1]
|
|
|
|
}
|
|
|
|
return rtn
|
|
|
|
}
|
|
|
|
// QuoteTypeDQ
|
|
|
|
return compQuoteDQString(s, close)
|
|
|
|
}
|
|
|
|
|
2022-11-10 05:38:28 +01:00
|
|
|
func (p *CompPoint) wordAsStr(w ParsedWord) string {
|
|
|
|
if w.Word != nil {
|
|
|
|
return p.StmtStr[w.Word.Pos().Offset():w.Word.End().Offset()]
|
|
|
|
}
|
|
|
|
return w.PartialWord
|
|
|
|
}
|
|
|
|
|
2022-11-22 09:26:41 +01:00
|
|
|
func (p *CompPoint) simpleExpandWord(w ParsedWord) (string, simpleexpand.SimpleExpandInfo) {
|
|
|
|
ectx := simpleexpand.SimpleExpandContext{}
|
2022-11-10 05:38:28 +01:00
|
|
|
if w.Word != nil {
|
2022-11-22 09:26:41 +01:00
|
|
|
return simpleexpand.SimpleExpandWord(ectx, w.Word, p.StmtStr)
|
2022-11-10 05:38:28 +01:00
|
|
|
}
|
2022-11-22 09:26:41 +01:00
|
|
|
return simpleexpand.SimpleExpandPartialWord(ectx, w.PartialWord, false)
|
2022-11-10 05:38:28 +01:00
|
|
|
}
|
|
|
|
|
2022-11-10 22:52:51 +01:00
|
|
|
func getQuoteTypePref(str string) string {
|
|
|
|
if strings.HasPrefix(str, QuoteTypeANSI) {
|
|
|
|
return QuoteTypeANSI
|
|
|
|
}
|
|
|
|
if strings.HasPrefix(str, QuoteTypeDQ) {
|
|
|
|
return QuoteTypeDQ
|
|
|
|
}
|
|
|
|
if strings.HasPrefix(str, QuoteTypeSQ) {
|
|
|
|
return QuoteTypeSQ
|
|
|
|
}
|
|
|
|
return QuoteTypeLiteral
|
|
|
|
}
|
|
|
|
|
2022-11-22 09:26:41 +01:00
|
|
|
func (p *CompPoint) getCompPrefix() (string, simpleexpand.SimpleExpandInfo) {
|
2022-11-10 05:38:28 +01:00
|
|
|
if p.CompWordPos == 0 {
|
2022-11-22 09:26:41 +01:00
|
|
|
return "", simpleexpand.SimpleExpandInfo{}
|
2022-11-10 05:38:28 +01:00
|
|
|
}
|
2022-11-10 22:52:51 +01:00
|
|
|
pword := p.Words[p.CompWord]
|
2022-11-10 05:38:28 +01:00
|
|
|
wordStr := p.wordAsStr(pword)
|
|
|
|
if p.CompWordPos == len(wordStr) {
|
|
|
|
return p.simpleExpandWord(pword)
|
|
|
|
}
|
|
|
|
// TODO we can do better, if p.Word is not nil, we can look for which WordPart
|
|
|
|
// our pos is in. we can then do a normal word expand on the previous parts
|
|
|
|
// and a partial on just the current part. this is an uncommon case though
|
|
|
|
// and has very little upside (even bash does not expand multipart words correctly)
|
|
|
|
partialWordStr := wordStr[:p.CompWordPos]
|
2022-11-22 09:26:41 +01:00
|
|
|
return simpleexpand.SimpleExpandPartialWord(simpleexpand.SimpleExpandContext{}, partialWordStr, false)
|
2022-11-10 22:52:51 +01:00
|
|
|
}
|
|
|
|
|
2022-11-22 09:26:41 +01:00
|
|
|
func (p *CompPoint) extendWord(newWord string, newWordComplete bool) utilfn.StrWithPos {
|
2022-11-10 22:52:51 +01:00
|
|
|
pword := p.Words[p.CompWord]
|
|
|
|
wordStr := p.wordAsStr(pword)
|
|
|
|
quotePref := getQuoteTypePref(wordStr)
|
|
|
|
needsClose := newWordComplete && (len(wordStr) == p.CompWordPos)
|
|
|
|
wordSuffix := wordStr[p.CompWordPos:]
|
|
|
|
newQuotedStr := compQuoteString(newWord, quotePref, needsClose)
|
2022-11-11 00:28:39 +01:00
|
|
|
if needsClose && wordSuffix == "" && !strings.HasSuffix(newWord, "/") {
|
2022-11-10 22:52:51 +01:00
|
|
|
newQuotedStr = newQuotedStr + " "
|
|
|
|
}
|
|
|
|
newPos := len(newQuotedStr)
|
2022-11-22 09:26:41 +01:00
|
|
|
return utilfn.StrWithPos{Str: newQuotedStr + wordSuffix, Pos: newPos}
|
|
|
|
}
|
|
|
|
|
|
|
|
// returns (extension, complete)
|
|
|
|
func computeCompExtension(compPrefix string, crtn *CompReturn) (string, bool) {
|
|
|
|
if crtn == nil || crtn.HasMore {
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
compStrs := crtn.GetCompStrs()
|
|
|
|
lcp := utilfn.LongestPrefix(compPrefix, compStrs)
|
|
|
|
if lcp == compPrefix || len(lcp) < len(compPrefix) || !strings.HasPrefix(lcp, compPrefix) {
|
|
|
|
return "", false
|
|
|
|
}
|
2022-11-22 09:32:27 +01:00
|
|
|
return lcp[len(compPrefix):], (utilfn.ContainsStr(compStrs, lcp) && !utilfn.IsPrefix(compStrs, lcp))
|
2022-11-11 00:28:39 +01:00
|
|
|
}
|
|
|
|
|
2022-11-22 09:26:41 +01:00
|
|
|
func (p *CompPoint) FullyExtend(crtn *CompReturn) utilfn.StrWithPos {
|
2022-11-11 00:28:39 +01:00
|
|
|
if crtn == nil || crtn.HasMore {
|
2022-11-22 09:26:41 +01:00
|
|
|
return utilfn.StrWithPos{Str: p.getOrigStr(), Pos: p.getOrigPos()}
|
2022-11-11 00:28:39 +01:00
|
|
|
}
|
2022-11-11 03:51:20 +01:00
|
|
|
compStrs := crtn.GetCompStrs()
|
2022-11-22 09:26:41 +01:00
|
|
|
compPrefix, _ := p.getCompPrefix()
|
2022-11-11 00:28:39 +01:00
|
|
|
lcp := utilfn.LongestPrefix(compPrefix, compStrs)
|
|
|
|
if lcp == compPrefix || len(lcp) < len(compPrefix) || !strings.HasPrefix(lcp, compPrefix) {
|
2022-11-22 09:26:41 +01:00
|
|
|
return utilfn.StrWithPos{Str: p.getOrigStr(), Pos: p.getOrigPos()}
|
2022-11-11 00:28:39 +01:00
|
|
|
}
|
|
|
|
newStr := p.extendWord(lcp, utilfn.ContainsStr(compStrs, lcp))
|
|
|
|
var buf bytes.Buffer
|
|
|
|
buf.WriteString(p.Prefix)
|
|
|
|
for idx, w := range p.Words {
|
|
|
|
if idx == p.CompWord {
|
|
|
|
buf.WriteString(w.Prefix)
|
|
|
|
buf.WriteString(newStr.Str)
|
|
|
|
} else {
|
|
|
|
buf.WriteString(w.Prefix)
|
|
|
|
buf.WriteString(p.wordAsStr(w))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
buf.WriteString(p.Suffix)
|
|
|
|
compWord := p.Words[p.CompWord]
|
|
|
|
newPos := len(p.Prefix) + compWord.Offset + len(compWord.Prefix) + newStr.Pos
|
2022-11-22 09:26:41 +01:00
|
|
|
return utilfn.StrWithPos{Str: buf.String(), Pos: newPos}
|
2022-11-10 05:38:28 +01:00
|
|
|
}
|
|
|
|
|
2022-11-04 03:16:14 +01:00
|
|
|
func (p *CompPoint) dump() {
|
|
|
|
if p.Prefix != "" {
|
|
|
|
fmt.Printf("prefix: %s\n", p.Prefix)
|
|
|
|
}
|
|
|
|
fmt.Printf("cpos: %d %d\n", p.CompWord, p.CompWordPos)
|
|
|
|
for idx, w := range p.Words {
|
|
|
|
fmt.Printf("w[%d]: ", idx)
|
|
|
|
if w.Prefix != "" {
|
|
|
|
fmt.Printf("{%s}", w.Prefix)
|
|
|
|
}
|
|
|
|
if idx == p.CompWord {
|
2022-11-22 09:26:41 +01:00
|
|
|
fmt.Printf("%s\n", utilfn.StrWithPos{Str: p.wordAsStr(w), Pos: p.CompWordPos})
|
2022-11-04 03:16:14 +01:00
|
|
|
} else {
|
2022-11-10 05:38:28 +01:00
|
|
|
fmt.Printf("%s\n", p.wordAsStr(w))
|
2022-11-04 03:16:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if p.Suffix != "" {
|
|
|
|
fmt.Printf("suffix: %s\n", p.Suffix)
|
|
|
|
}
|
|
|
|
fmt.Printf("\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
var SimpleCompGenFns map[string]SimpleCompGenFnType
|
|
|
|
|
|
|
|
func isWhitespace(str string) bool {
|
|
|
|
return strings.TrimSpace(str) == ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func splitInitialWhitespace(str string) (string, string) {
|
|
|
|
for pos, ch := range str { // rune iteration :/
|
|
|
|
if !unicode.IsSpace(ch) {
|
|
|
|
return str[:pos], str[pos:]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return str, ""
|
|
|
|
}
|
|
|
|
|
2022-11-22 09:26:41 +01:00
|
|
|
func ParseCompPoint(cmdStr utilfn.StrWithPos) *CompPoint {
|
2022-11-11 01:00:51 +01:00
|
|
|
fullCmdStr := cmdStr.Str
|
|
|
|
pos := cmdStr.Pos
|
2022-11-04 03:16:14 +01:00
|
|
|
// fmt.Printf("---\n")
|
|
|
|
// fmt.Printf("cmd: %s\n", strWithCursor(fullCmdStr, pos))
|
|
|
|
|
|
|
|
// first, find the stmt that the pos appears in
|
|
|
|
cmdReader := strings.NewReader(fullCmdStr)
|
|
|
|
parser := syntax.NewParser(syntax.Variant(syntax.LangBash))
|
|
|
|
var foundStmt *syntax.Stmt
|
|
|
|
var lastStmt *syntax.Stmt
|
|
|
|
var restStartPos int
|
|
|
|
parser.Stmts(cmdReader, func(stmt *syntax.Stmt) bool { // ignore parse errors (since stmtStr will be the unparsed part)
|
|
|
|
restStartPos = int(stmt.End().Offset())
|
|
|
|
lastStmt = stmt
|
|
|
|
if uint(pos) >= stmt.Pos().Offset() && uint(pos) < stmt.End().Offset() {
|
|
|
|
foundStmt = stmt
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
// fmt.Printf("stmt: [[%s]] %d:%d (%d)\n", fullCmdStr[stmt.Pos().Offset():stmt.End().Offset()], stmt.Pos().Offset(), stmt.End().Offset(), stmt.Semicolon.Offset())
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
restStr := fullCmdStr[restStartPos:]
|
|
|
|
if foundStmt == nil && lastStmt != nil && isWhitespace(restStr) && lastStmt.Semicolon.Offset() == 0 {
|
|
|
|
foundStmt = lastStmt
|
|
|
|
}
|
|
|
|
var rtnPoint CompPoint
|
|
|
|
var stmtStr string
|
|
|
|
var stmtPos int
|
|
|
|
if foundStmt != nil {
|
|
|
|
stmtPos = pos - int(foundStmt.Pos().Offset())
|
|
|
|
rtnPoint.Prefix = fullCmdStr[:foundStmt.Pos().Offset()]
|
|
|
|
if isWhitespace(fullCmdStr[foundStmt.End().Offset():]) {
|
|
|
|
stmtStr = fullCmdStr[foundStmt.Pos().Offset():]
|
|
|
|
rtnPoint.Suffix = ""
|
|
|
|
} else {
|
|
|
|
stmtStr = fullCmdStr[foundStmt.Pos().Offset():foundStmt.End().Offset()]
|
|
|
|
rtnPoint.Suffix = fullCmdStr[foundStmt.End().Offset():]
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
stmtStr = restStr
|
|
|
|
stmtPos = pos - restStartPos
|
|
|
|
rtnPoint.Prefix = fullCmdStr[:restStartPos]
|
|
|
|
rtnPoint.Suffix = fullCmdStr[restStartPos+len(stmtStr):]
|
|
|
|
}
|
|
|
|
if stmtPos > len(stmtStr) {
|
|
|
|
// this should not happen and will cause a jump in completed strings
|
|
|
|
stmtPos = len(stmtStr)
|
|
|
|
}
|
|
|
|
// fmt.Printf("found: ((%s))%s((%s))\n", rtnPoint.Prefix, strWithCursor(stmtStr, stmtPos), rtnPoint.Suffix)
|
|
|
|
|
|
|
|
// now, find the word that the pos appears in within the stmt above
|
2022-11-10 05:38:28 +01:00
|
|
|
rtnPoint.StmtStr = stmtStr
|
2022-11-04 03:16:14 +01:00
|
|
|
stmtReader := strings.NewReader(stmtStr)
|
|
|
|
lastWordPos := 0
|
|
|
|
parser.Words(stmtReader, func(w *syntax.Word) bool {
|
|
|
|
var pword ParsedWord
|
|
|
|
pword.Offset = lastWordPos
|
|
|
|
if int(w.Pos().Offset()) > lastWordPos {
|
|
|
|
pword.Prefix = stmtStr[lastWordPos:w.Pos().Offset()]
|
|
|
|
}
|
2022-11-10 05:38:28 +01:00
|
|
|
pword.Word = w
|
2022-11-04 03:16:14 +01:00
|
|
|
rtnPoint.Words = append(rtnPoint.Words, pword)
|
|
|
|
lastWordPos = int(w.End().Offset())
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
if lastWordPos < len(stmtStr) {
|
|
|
|
pword := ParsedWord{Offset: lastWordPos}
|
2022-11-10 05:38:28 +01:00
|
|
|
pword.Prefix, pword.PartialWord = splitInitialWhitespace(stmtStr[lastWordPos:])
|
2022-11-04 03:16:14 +01:00
|
|
|
rtnPoint.Words = append(rtnPoint.Words, pword)
|
|
|
|
}
|
|
|
|
if len(rtnPoint.Words) == 0 {
|
|
|
|
rtnPoint.Words = append(rtnPoint.Words, ParsedWord{})
|
|
|
|
}
|
|
|
|
for idx, w := range rtnPoint.Words {
|
2022-11-10 05:38:28 +01:00
|
|
|
wordLen := len(rtnPoint.wordAsStr(w))
|
|
|
|
if stmtPos > w.Offset && stmtPos <= w.Offset+len(w.Prefix)+wordLen {
|
2022-11-04 03:16:14 +01:00
|
|
|
rtnPoint.CompWord = idx
|
|
|
|
rtnPoint.CompWordPos = stmtPos - w.Offset - len(w.Prefix)
|
|
|
|
if rtnPoint.CompWordPos < 0 {
|
|
|
|
splitCompWord(&rtnPoint)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-11-11 01:00:51 +01:00
|
|
|
return &rtnPoint
|
2022-11-04 03:16:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func splitCompWord(p *CompPoint) {
|
|
|
|
w := p.Words[p.CompWord]
|
|
|
|
prefixPos := p.CompWordPos + len(w.Prefix)
|
|
|
|
|
2022-11-10 05:38:28 +01:00
|
|
|
w1 := ParsedWord{Offset: w.Offset, Prefix: w.Prefix[:prefixPos]}
|
|
|
|
w2 := ParsedWord{Offset: w.Offset + prefixPos, Prefix: w.Prefix[prefixPos:], Word: w.Word, PartialWord: w.PartialWord}
|
2022-11-04 03:16:14 +01:00
|
|
|
p.CompWord = p.CompWord // the same (w1)
|
|
|
|
p.CompWordPos = 0 // will be at 0 since w1 has a word length of 0
|
|
|
|
var newWords []ParsedWord
|
|
|
|
if p.CompWord > 0 {
|
|
|
|
newWords = append(newWords, p.Words[0:p.CompWord]...)
|
|
|
|
}
|
|
|
|
newWords = append(newWords, w1, w2)
|
|
|
|
newWords = append(newWords, p.Words[p.CompWord+1:]...)
|
|
|
|
p.Words = newWords
|
|
|
|
}
|
|
|
|
|
2022-11-22 09:26:41 +01:00
|
|
|
func getCompType(compPos shparse.CompletionPos) string {
|
|
|
|
switch compPos.CompType {
|
|
|
|
case shparse.CompTypeCommandMeta:
|
|
|
|
return CGTypeCommandMeta
|
|
|
|
|
|
|
|
case shparse.CompTypeCommand:
|
|
|
|
return CGTypeCommand
|
|
|
|
|
|
|
|
case shparse.CompTypeVar:
|
|
|
|
return CGTypeVariable
|
|
|
|
|
|
|
|
case shparse.CompTypeArg, shparse.CompTypeBasic, shparse.CompTypeAssignment:
|
|
|
|
return CGTypeFile
|
|
|
|
|
|
|
|
default:
|
|
|
|
return CGTypeFile
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func fixupVarPrefix(varPrefix string) string {
|
|
|
|
if strings.HasPrefix(varPrefix, "${") {
|
|
|
|
varPrefix = varPrefix[2:]
|
|
|
|
if strings.HasSuffix(varPrefix, "}") {
|
|
|
|
varPrefix = varPrefix[:len(varPrefix)-1]
|
|
|
|
}
|
|
|
|
} else if strings.HasPrefix(varPrefix, "$") {
|
|
|
|
varPrefix = varPrefix[1:]
|
|
|
|
}
|
|
|
|
return varPrefix
|
|
|
|
}
|
|
|
|
|
|
|
|
func DoCompGen(ctx context.Context, cmdStr utilfn.StrWithPos, compCtx CompContext) (*CompReturn, *utilfn.StrWithPos, error) {
|
|
|
|
words := shparse.Tokenize(cmdStr.Str)
|
|
|
|
cmds := shparse.ParseCommands(words)
|
2022-11-23 23:34:49 +01:00
|
|
|
compPos := shparse.FindCompletionPos(cmds, cmdStr.Pos)
|
2022-11-22 09:26:41 +01:00
|
|
|
if compPos.CompType == shparse.CompTypeInvalid {
|
|
|
|
return nil, nil, nil
|
|
|
|
}
|
|
|
|
var compPrefix string
|
|
|
|
if compPos.CompWord != nil {
|
|
|
|
var info shparse.ExpandInfo
|
|
|
|
compPrefix, info = shparse.SimpleExpandPrefix(shparse.ExpandContext{}, compPos.CompWord, compPos.CompWordOffset)
|
|
|
|
if info.HasGlob || info.HasExtGlob || info.HasHistory || info.HasSpecial {
|
|
|
|
return nil, nil, nil
|
|
|
|
}
|
|
|
|
if compPos.CompType != shparse.CompTypeVar && info.HasVar {
|
|
|
|
return nil, nil, nil
|
|
|
|
}
|
|
|
|
if compPos.CompType == shparse.CompTypeVar {
|
|
|
|
compPrefix = fixupVarPrefix(compPrefix)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
scType := getCompType(compPos)
|
|
|
|
crtn, err := DoSimpleComp(ctx, scType, compPrefix, compCtx, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
if compCtx.ForDisplay {
|
|
|
|
return crtn, nil, nil
|
|
|
|
}
|
|
|
|
extensionStr, extensionComplete := computeCompExtension(compPrefix, crtn)
|
|
|
|
if extensionStr == "" {
|
|
|
|
return crtn, nil, nil
|
|
|
|
}
|
|
|
|
rtnSP := compPos.Extend(cmdStr, extensionStr, extensionComplete)
|
|
|
|
return crtn, &rtnSP, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func DoCompGenOld(ctx context.Context, sp utilfn.StrWithPos, compCtx CompContext) (*CompReturn, *utilfn.StrWithPos, error) {
|
2022-11-11 01:00:51 +01:00
|
|
|
compPoint := ParseCompPoint(sp)
|
|
|
|
compType := CGTypeFile
|
|
|
|
if compPoint.CompWord == 0 {
|
|
|
|
compType = CGTypeCommandMeta
|
|
|
|
}
|
|
|
|
// TODO lookup special types
|
2022-11-22 09:26:41 +01:00
|
|
|
compPrefix, info := compPoint.getCompPrefix()
|
|
|
|
if info.HasVar || info.HasGlob || info.HasExtGlob || info.HasHistory || info.HasSpecial {
|
|
|
|
return nil, nil, nil
|
|
|
|
}
|
2022-11-11 01:00:51 +01:00
|
|
|
crtn, err := DoSimpleComp(ctx, compType, compPrefix, compCtx, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
if compCtx.ForDisplay {
|
|
|
|
return crtn, nil, nil
|
|
|
|
}
|
|
|
|
rtnSP := compPoint.FullyExtend(crtn)
|
|
|
|
return crtn, &rtnSP, nil
|
2022-11-04 03:16:14 +01:00
|
|
|
}
|
2022-11-10 22:52:51 +01:00
|
|
|
|
|
|
|
func SortCompReturnEntries(c *CompReturn) {
|
|
|
|
sort.Slice(c.Entries, func(i int, j int) bool {
|
|
|
|
e1 := c.Entries[i]
|
|
|
|
e2 := c.Entries[j]
|
|
|
|
if e1.Word < e2.Word {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if e1.Word == e2.Word && e1.IsMetaCmd && !e2.IsMetaCmd {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-11-11 03:51:20 +01:00
|
|
|
func CombineCompReturn(compType string, c1 *CompReturn, c2 *CompReturn) *CompReturn {
|
2022-11-10 22:52:51 +01:00
|
|
|
if c1 == nil {
|
|
|
|
return c2
|
|
|
|
}
|
|
|
|
if c2 == nil {
|
|
|
|
return c1
|
|
|
|
}
|
|
|
|
var rtn CompReturn
|
2022-11-11 03:51:20 +01:00
|
|
|
rtn.CompType = compType
|
2022-11-10 22:52:51 +01:00
|
|
|
rtn.HasMore = c1.HasMore || c2.HasMore
|
|
|
|
rtn.Entries = append([]CompEntry{}, c1.Entries...)
|
|
|
|
rtn.Entries = append(rtn.Entries, c2.Entries...)
|
|
|
|
SortCompReturnEntries(&rtn)
|
|
|
|
return &rtn
|
|
|
|
}
|
2022-11-11 00:28:39 +01:00
|
|
|
|
2022-11-11 03:51:20 +01:00
|
|
|
func (c *CompReturn) GetCompStrs() []string {
|
2022-11-11 00:28:39 +01:00
|
|
|
rtn := make([]string, len(c.Entries))
|
|
|
|
for idx, entry := range c.Entries {
|
|
|
|
rtn[idx] = entry.Word
|
|
|
|
}
|
|
|
|
return rtn
|
|
|
|
}
|
|
|
|
|
2022-11-11 03:51:20 +01:00
|
|
|
func (c *CompReturn) GetCompDisplayStrs() []string {
|
|
|
|
rtn := make([]string, len(c.Entries))
|
|
|
|
for idx, entry := range c.Entries {
|
|
|
|
if entry.IsMetaCmd {
|
|
|
|
rtn[idx] = "^" + entry.Word
|
|
|
|
} else {
|
|
|
|
rtn[idx] = entry.Word
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rtn
|
|
|
|
}
|
|
|
|
|
2022-11-11 00:28:39 +01:00
|
|
|
func (p CompPoint) getOrigPos() int {
|
|
|
|
pword := p.Words[p.CompWord]
|
|
|
|
return len(p.Prefix) + pword.Offset + len(pword.Prefix) + p.CompWordPos
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p CompPoint) getOrigStr() string {
|
|
|
|
return p.Prefix + p.StmtStr + p.Suffix
|
|
|
|
}
|