mirror of https://github.com/status-im/op-geth.git
cmd/geth, jsre: improve auto-completion
This commit is contained in:
parent
2abf1a36b9
commit
ae5bc89cad
|
@ -80,51 +80,11 @@ type jsre struct {
|
||||||
prompter
|
prompter
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
func makeCompleter(re *jsre) liner.WordCompleter {
|
||||||
loadedModulesMethods map[string][]string
|
return func(line string, pos int) (head string, completions []string, tail string) {
|
||||||
autoCompleteStatement = "function _autocomplete(obj) {var results = []; for (var e in obj) { results.push(e); }; return results; }; _autocomplete(%s)"
|
|
||||||
)
|
|
||||||
|
|
||||||
func keywordCompleter(jsre *jsre, line string) []string {
|
|
||||||
var results []string
|
|
||||||
parts := strings.Split(line, ".")
|
|
||||||
objRef := "this"
|
|
||||||
prefix := line
|
|
||||||
if len(parts) > 1 {
|
|
||||||
objRef = strings.Join(parts[0:len(parts)-1], ".")
|
|
||||||
prefix = parts[len(parts)-1]
|
|
||||||
}
|
|
||||||
|
|
||||||
result, _ := jsre.re.Run(fmt.Sprintf(autoCompleteStatement, objRef))
|
|
||||||
raw, _ := result.Export()
|
|
||||||
if keys, ok := raw.([]interface{}); ok {
|
|
||||||
for _, k := range keys {
|
|
||||||
if strings.HasPrefix(fmt.Sprintf("%s", k), prefix) {
|
|
||||||
if objRef == "this" {
|
|
||||||
results = append(results, fmt.Sprintf("%s", k))
|
|
||||||
} else {
|
|
||||||
results = append(results, fmt.Sprintf("%s.%s", strings.Join(parts[:len(parts)-1], "."), k))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// e.g. web3<tab><tab> append dot since its an object
|
|
||||||
isObj, _ := jsre.re.Run(fmt.Sprintf("typeof(%s) === 'object'", line))
|
|
||||||
if isObject, _ := isObj.ToBoolean(); isObject {
|
|
||||||
results = append(results, line+".")
|
|
||||||
}
|
|
||||||
|
|
||||||
sort.Strings(results)
|
|
||||||
return results
|
|
||||||
}
|
|
||||||
|
|
||||||
func apiWordCompleterWithContext(jsre *jsre) liner.WordCompleter {
|
|
||||||
completer := func(line string, pos int) (head string, completions []string, tail string) {
|
|
||||||
if len(line) == 0 || pos == 0 {
|
if len(line) == 0 || pos == 0 {
|
||||||
return "", nil, ""
|
return "", nil, ""
|
||||||
}
|
}
|
||||||
|
|
||||||
// chuck data to relevant part for autocompletion, e.g. in case of nested lines eth.getBalance(eth.coinb<tab><tab>
|
// chuck data to relevant part for autocompletion, e.g. in case of nested lines eth.getBalance(eth.coinb<tab><tab>
|
||||||
i := 0
|
i := 0
|
||||||
for i = pos - 1; i > 0; i-- {
|
for i = pos - 1; i > 0; i-- {
|
||||||
|
@ -137,16 +97,8 @@ func apiWordCompleterWithContext(jsre *jsre) liner.WordCompleter {
|
||||||
i += 1
|
i += 1
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
return line[:i], re.re.CompleteKeywords(line[i:pos]), line[pos:]
|
||||||
begin := line[:i]
|
|
||||||
keyword := line[i:pos]
|
|
||||||
end := line[pos:]
|
|
||||||
|
|
||||||
completionWords := keywordCompleter(jsre, keyword)
|
|
||||||
return begin, completionWords, end
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return completer
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func newLightweightJSRE(docRoot string, client rpc.Client, datadir string, interactive bool) *jsre {
|
func newLightweightJSRE(docRoot string, client rpc.Client, datadir string, interactive bool) *jsre {
|
||||||
|
@ -165,8 +117,7 @@ func newLightweightJSRE(docRoot string, client rpc.Client, datadir string, inter
|
||||||
lr := liner.NewLiner()
|
lr := liner.NewLiner()
|
||||||
js.withHistory(datadir, func(hist *os.File) { lr.ReadHistory(hist) })
|
js.withHistory(datadir, func(hist *os.File) { lr.ReadHistory(hist) })
|
||||||
lr.SetCtrlCAborts(true)
|
lr.SetCtrlCAborts(true)
|
||||||
js.loadAutoCompletion()
|
lr.SetWordCompleter(makeCompleter(js))
|
||||||
lr.SetWordCompleter(apiWordCompleterWithContext(js))
|
|
||||||
lr.SetTabCompletionStyle(liner.TabPrints)
|
lr.SetTabCompletionStyle(liner.TabPrints)
|
||||||
lr.SetMultiLineMode(true)
|
lr.SetMultiLineMode(true)
|
||||||
js.prompter = lr
|
js.prompter = lr
|
||||||
|
@ -197,8 +148,7 @@ func newJSRE(stack *node.Node, docRoot, corsDomain string, client rpc.Client, in
|
||||||
lr := liner.NewLiner()
|
lr := liner.NewLiner()
|
||||||
js.withHistory(stack.DataDir(), func(hist *os.File) { lr.ReadHistory(hist) })
|
js.withHistory(stack.DataDir(), func(hist *os.File) { lr.ReadHistory(hist) })
|
||||||
lr.SetCtrlCAborts(true)
|
lr.SetCtrlCAborts(true)
|
||||||
js.loadAutoCompletion()
|
lr.SetWordCompleter(makeCompleter(js))
|
||||||
lr.SetWordCompleter(apiWordCompleterWithContext(js))
|
|
||||||
lr.SetTabCompletionStyle(liner.TabPrints)
|
lr.SetTabCompletionStyle(liner.TabPrints)
|
||||||
js.prompter = lr
|
js.prompter = lr
|
||||||
js.atexit = func() {
|
js.atexit = func() {
|
||||||
|
@ -210,15 +160,6 @@ func newJSRE(stack *node.Node, docRoot, corsDomain string, client rpc.Client, in
|
||||||
return js
|
return js
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *jsre) loadAutoCompletion() {
|
|
||||||
if modules, err := self.supportedApis(); err == nil {
|
|
||||||
loadedModulesMethods = make(map[string][]string)
|
|
||||||
for module, _ := range modules {
|
|
||||||
loadedModulesMethods[module] = rpc.AutoCompletion[module]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (self *jsre) batch(statement string) {
|
func (self *jsre) batch(statement string) {
|
||||||
err := self.re.EvalAndPrettyPrint(statement)
|
err := self.re.EvalAndPrettyPrint(statement)
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,63 @@
|
||||||
|
// Copyright 2015 The go-ethereum Authors
|
||||||
|
// This file is part of the go-ethereum library.
|
||||||
|
//
|
||||||
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU Lesser General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU Lesser General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU Lesser General Public License
|
||||||
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
package jsre
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/robertkrimen/otto"
|
||||||
|
)
|
||||||
|
|
||||||
|
// CompleteKeywords returns potential continuations for the given line. Since line is
|
||||||
|
// evaluated, callers need to make sure that evaluating line does not have side effects.
|
||||||
|
func (jsre *JSRE) CompleteKeywords(line string) []string {
|
||||||
|
var results []string
|
||||||
|
jsre.do(func(vm *otto.Otto) { results = getCompletions(vm, line) })
|
||||||
|
return results
|
||||||
|
}
|
||||||
|
|
||||||
|
func getCompletions(vm *otto.Otto, line string) (results []string) {
|
||||||
|
parts := strings.Split(line, ".")
|
||||||
|
objRef := "this"
|
||||||
|
prefix := line
|
||||||
|
if len(parts) > 1 {
|
||||||
|
objRef = strings.Join(parts[0:len(parts)-1], ".")
|
||||||
|
prefix = parts[len(parts)-1]
|
||||||
|
}
|
||||||
|
|
||||||
|
res, err := vm.Eval(objRef)
|
||||||
|
if err != nil || !res.IsObject() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
for _, k := range res.Object().Keys() {
|
||||||
|
if strings.HasPrefix(k, prefix) {
|
||||||
|
if objRef == "this" {
|
||||||
|
results = append(results, k)
|
||||||
|
} else {
|
||||||
|
results = append(results, strings.Join(parts[:len(parts)-1], ".")+"."+k)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// e.g. web3<tab><tab> append dot since its an object
|
||||||
|
if lineRes, _ := vm.Eval(line); lineRes.IsObject() {
|
||||||
|
results = append(results, line+".")
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.Strings(results)
|
||||||
|
return results
|
||||||
|
}
|
Loading…
Reference in New Issue