Skip to content

Commit 50d05eb

Browse files
authored
Initial integration with fortio.org/terminal (#120)
* initial integration with fortio.org/terminal * eof/^c aren't errors, for now use compact form to add to history * add history, !n, and help commands to interactive mode * add info about history interactive repl feature and max-history flag * terminal with fix for -logger-no-color * cleanup eof code, no setprompt needed, done by terminal/ now * use os.UserConfigDir for now but I don't like it * Switch back to home dir but add GROL_HISTORY_FILE env option, also use home dir as a virtual '~/' that gets replaced by actual if unchanged from default, so pasting help doesn't leak the home dir * use terminal 0.7.0, term.LoggerSetup is automatic * Use the constant from the package as default * Fix up the go.mod
1 parent f9b03a6 commit 50d05eb

File tree

5 files changed

+142
-40
lines changed

5 files changed

+142
-40
lines changed

README.md

+10-1
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,9 @@ $ info.keywords
6565
== Eval ==> ["else","error","false","first","func","if","len","log","macro","print","println","quote","rest","return","true","unquote"]
6666
```
6767

68+
There is also in interactive repl mode: `history`, `!23` to repeat the 23rd statement for instance and `help`.
69+
And full edit and history navigation with arrow keys etc...
70+
6871
## Language features
6972

7073
Functional int, float, string and boolean expressions
@@ -167,7 +170,7 @@ See [Open Issues](https://grol.io/grol/issues) for what's left to do
167170
### CLI Usage
168171

169172
```
170-
grol 0.29.0 usage:
173+
grol 0.38.0 usage:
171174
grol [flags] *.gr files to interpret or `-` for stdin without prompt
172175
or no arguments for stdin repl...
173176
or 1 of the special arguments
@@ -181,9 +184,15 @@ flags:
181184
show eval results (default true)
182185
-format
183186
don't execute, just parse and re format the input
187+
-history string
188+
history file to use (default "~/.grol_history")
189+
-max-history size
190+
max history size, use 0 to disable. (default 99)
184191
-parse
185192
show parse tree
186193
-shared-state
187194
All files share same interpreter state (default is new state for each)
188195
```
189196
(excluding logger control, see `gorepl help` for all the flags, of note `-logger-no-color` will turn off colors for gorepl too, for development there are also `-profile*` options for pprof, when building without `no_pprof`)
197+
198+
If you don't want to pass a flag and want to permanently change the `grol` history file location from your HOME directory, set GROL_HISTORY_FILE in the environment.

go.mod

+7-5
Original file line numberDiff line numberDiff line change
@@ -6,15 +6,17 @@ require (
66
fortio.org/cli v1.8.0
77
fortio.org/log v1.16.0
88
fortio.org/sets v1.2.0
9+
fortio.org/struct2env v0.4.1
10+
fortio.org/terminal v0.7.2
911
fortio.org/testscript v0.3.1 // only for tests
1012
fortio.org/version v1.0.4
1113
)
1214

1315
require (
14-
fortio.org/struct2env v0.4.1 // indirect
16+
fortio.org/term v0.23.0-fortio-6 // indirect
1517
github.com/kortschak/goroutine v1.1.2 // indirect
16-
golang.org/x/crypto/x509roots/fallback v0.0.0-20240626151235-a6a393ffd658 // indirect
17-
golang.org/x/exp v0.0.0-20240719175910-8a7402abbf56 // indirect
18-
golang.org/x/sys v0.22.0 // indirect
19-
golang.org/x/tools v0.23.0 // indirect
18+
golang.org/x/crypto/x509roots/fallback v0.0.0-20240806160748-b2d3a6a4b4d3 // indirect
19+
golang.org/x/exp v0.0.0-20240808152545-0cdaa3abc0fa // indirect
20+
golang.org/x/sys v0.24.0 // indirect
21+
golang.org/x/tools v0.24.0 // indirect
2022
)

go.sum

+12-8
Original file line numberDiff line numberDiff line change
@@ -8,17 +8,21 @@ fortio.org/sets v1.2.0 h1:FBfC7R2xrOJtkcioUbY6WqEzdujuBoZRbSdp1fYF4Kk=
88
fortio.org/sets v1.2.0/go.mod h1:J2BwIxNOLWsSU7IMZUg541kh3Au4JEKHrghVwXs68tE=
99
fortio.org/struct2env v0.4.1 h1:rJludAMO5eBvpWplWEQNqoVDFZr4RWMQX7RUapgZyc0=
1010
fortio.org/struct2env v0.4.1/go.mod h1:lENUe70UwA1zDUCX+8AsO663QCFqYaprk5lnPhjD410=
11+
fortio.org/term v0.23.0-fortio-6 h1:pKrUX0tKOxyEhkhLV50oJYucTVx94rzFrXc24lIuLvk=
12+
fortio.org/term v0.23.0-fortio-6/go.mod h1:7buBfn81wEJUGWiVjFNiUE/vxWs5FdM9c7PyZpZRS30=
13+
fortio.org/terminal v0.7.2 h1:Bfpw6ORqrpaTVtlP0NxSS2VyfESR17DvT9qqz75ixJU=
14+
fortio.org/terminal v0.7.2/go.mod h1:Z/dydQSo8hCwiUGOt2pJiR8OsNAFn6pTt3pbHDsdtSM=
1115
fortio.org/testscript v0.3.1 h1:MmRO64AsmzaU1KlYMzAbotJIMKRGxD1XXssJnBRiMGQ=
1216
fortio.org/testscript v0.3.1/go.mod h1:7OJ+U4avooRNqc7p/VHKJadYgj9fA6+N0SbGU8FVWGs=
1317
fortio.org/version v1.0.4 h1:FWUMpJ+hVTNc4RhvvOJzb0xesrlRmG/a+D6bjbQ4+5U=
1418
fortio.org/version v1.0.4/go.mod h1:2JQp9Ax+tm6QKiGuzR5nJY63kFeANcgrZ0osoQFDVm0=
1519
github.com/kortschak/goroutine v1.1.2 h1:lhllcCuERxMIK5cYr8yohZZScL1na+JM5JYPRclWjck=
1620
github.com/kortschak/goroutine v1.1.2/go.mod h1:zKpXs1FWN/6mXasDQzfl7g0LrGFIOiA6cLs9eXKyaMY=
17-
golang.org/x/crypto/x509roots/fallback v0.0.0-20240626151235-a6a393ffd658 h1:i7K6wQLN/0oxF7FT3tKkfMCstxoT4VGG36YIB9ZKLzI=
18-
golang.org/x/crypto/x509roots/fallback v0.0.0-20240626151235-a6a393ffd658/go.mod h1:kNa9WdvYnzFwC79zRpLRMJbdEFlhyM5RPFBBZp/wWH8=
19-
golang.org/x/exp v0.0.0-20240719175910-8a7402abbf56 h1:2dVuKD2vS7b0QIHQbpyTISPd0LeHDbnYEryqj5Q1ug8=
20-
golang.org/x/exp v0.0.0-20240719175910-8a7402abbf56/go.mod h1:M4RDyNAINzryxdtnbRXRL/OHtkFuWGRjvuhBJpk2IlY=
21-
golang.org/x/sys v0.22.0 h1:RI27ohtqKCnwULzJLqkv897zojh5/DwS/ENaMzUOaWI=
22-
golang.org/x/sys v0.22.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
23-
golang.org/x/tools v0.23.0 h1:SGsXPZ+2l4JsgaCKkx+FQ9YZ5XEtA1GZYuoDjenLjvg=
24-
golang.org/x/tools v0.23.0/go.mod h1:pnu6ufv6vQkll6szChhK3C3L/ruaIv5eBeztNG8wtsI=
21+
golang.org/x/crypto/x509roots/fallback v0.0.0-20240806160748-b2d3a6a4b4d3 h1:oWb21rU9Q9XrRwXLB7jHc1rbp6EiiimZZv5MLxpu4T0=
22+
golang.org/x/crypto/x509roots/fallback v0.0.0-20240806160748-b2d3a6a4b4d3/go.mod h1:kNa9WdvYnzFwC79zRpLRMJbdEFlhyM5RPFBBZp/wWH8=
23+
golang.org/x/exp v0.0.0-20240808152545-0cdaa3abc0fa h1:ELnwvuAXPNtPk1TJRuGkI9fDTwym6AYBu0qzT8AcHdI=
24+
golang.org/x/exp v0.0.0-20240808152545-0cdaa3abc0fa/go.mod h1:akd2r19cwCdwSwWeIdzYQGa/EZZyqcOdwWiwj5L5eKQ=
25+
golang.org/x/sys v0.24.0 h1:Twjiwq9dn6R1fQcyiK+wQyHWfaz/BJB+YIpzU/Cv3Xg=
26+
golang.org/x/sys v0.24.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
27+
golang.org/x/tools v0.24.0 h1:J1shsA93PJUEVaUSaay7UXAyE8aimq3GW0pjlolpa24=
28+
golang.org/x/tools v0.24.0/go.mod h1:YhNqVBIfWHdzvTLs0d8LCuMhkKUgSUKldakyV7W/WDQ=

main.go

+44-7
Original file line numberDiff line numberDiff line change
@@ -6,9 +6,12 @@ import (
66
"fmt"
77
"io"
88
"os"
9+
"path/filepath"
910

1011
"fortio.org/cli"
1112
"fortio.org/log"
13+
"fortio.org/struct2env"
14+
"fortio.org/terminal"
1215
"grol.io/grol/eval"
1316
"grol.io/grol/extensions" // register extensions
1417
"grol.io/grol/object"
@@ -19,6 +22,19 @@ func main() {
1922
os.Exit(Main())
2023
}
2124

25+
type Config struct {
26+
HistoryFile string
27+
}
28+
29+
var config = Config{}
30+
31+
func EnvHelp(w io.Writer) {
32+
res, _ := struct2env.StructToEnvVars(config)
33+
str := struct2env.ToShellWithPrefix("GROL_", res, true)
34+
fmt.Fprintln(w, "# Grol environment variables:")
35+
fmt.Fprint(w, str)
36+
}
37+
2238
var hookBefore, hookAfter func() int
2339

2440
func Main() int {
@@ -28,16 +44,38 @@ func Main() int {
2844
compact := flag.Bool("compact", false, "When printing code, use no indentation and most compact form")
2945
showEval := flag.Bool("eval", true, "show eval results")
3046
sharedState := flag.Bool("shared-state", false, "All files share same interpreter state (default is new state for each)")
31-
47+
const historyDefault = "~/.grol_history" // virtual/token filename, will be replaced by actual home dir if not changed.
48+
cli.EnvHelpFuncs = append(cli.EnvHelpFuncs, EnvHelp)
49+
defaultHistoryFile := historyDefault
50+
errs := struct2env.SetFromEnv("GROL_", &config)
51+
if len(errs) > 0 {
52+
log.Errf("Error setting config from env: %v", errs)
53+
}
54+
if config.HistoryFile != "" {
55+
defaultHistoryFile = config.HistoryFile
56+
}
57+
historyFile := flag.String("history", defaultHistoryFile, "history `file` to use")
58+
maxHistory := flag.Int("max-history", terminal.DefaultHistoryCapacity, "max history `size`, use 0 to disable.")
3259
cli.ArgsHelp = "*.gr files to interpret or `-` for stdin without prompt or no arguments for stdin repl..."
3360
cli.MaxArgs = -1
3461
cli.Main()
62+
histFile := *historyFile
63+
if histFile == historyDefault {
64+
homeDir, err := os.UserHomeDir()
65+
histFile = filepath.Join(homeDir, ".grol_history")
66+
if err != nil {
67+
log.Warnf("Couldn't get user home dir: %v", err)
68+
histFile = ""
69+
}
70+
}
3571
log.Infof("grol %s - welcome!", cli.LongVersion)
3672
options := repl.Options{
37-
ShowParse: *showParse,
38-
ShowEval: *showEval,
39-
FormatOnly: *format,
40-
Compact: *compact,
73+
ShowParse: *showParse,
74+
ShowEval: *showEval,
75+
FormatOnly: *format,
76+
Compact: *compact,
77+
HistoryFile: histFile,
78+
MaxHistory: *maxHistory,
4179
}
4280
if hookBefore != nil {
4381
ret := hookBefore()
@@ -58,8 +96,7 @@ func Main() int {
5896
return len(errs)
5997
}
6098
if len(flag.Args()) == 0 {
61-
repl.Interactive(os.Stdin, os.Stdout, options)
62-
return 0
99+
return repl.Interactive(options)
63100
}
64101
options.All = true
65102
s := eval.NewState()

repl/repl.go

+69-19
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,16 @@
11
package repl
22

33
import (
4-
"bufio"
4+
"errors"
55
"fmt"
66
"io"
7+
"regexp"
8+
"slices"
9+
"strconv"
710
"strings"
811

912
"fortio.org/log"
13+
"fortio.org/terminal"
1014
"grol.io/grol/ast"
1115
"grol.io/grol/eval"
1216
"grol.io/grol/lexer"
@@ -33,13 +37,15 @@ func logParserErrors(p *parser.Parser) bool {
3337
}
3438

3539
type Options struct {
36-
ShowParse bool
37-
ShowEval bool
38-
All bool
39-
NoColor bool // color controlled by log package, unless this is set to true.
40-
FormatOnly bool
41-
Compact bool
42-
NilAndErr bool // Show nil and errors in normal output.
40+
ShowParse bool
41+
ShowEval bool
42+
All bool
43+
NoColor bool // color controlled by log package, unless this is set to true.
44+
FormatOnly bool
45+
Compact bool
46+
NilAndErr bool // Show nil and errors in normal output.
47+
HistoryFile string
48+
MaxHistory int
4349
}
4450

4551
func EvalAll(s *eval.State, macroState *object.Environment, in io.Reader, out io.Writer, options Options) []string {
@@ -78,29 +84,73 @@ func EvalString(what string) (res string, errs []string, formatted string) {
7884
return
7985
}
8086

81-
func Interactive(in io.Reader, out io.Writer, options Options) {
87+
func Interactive(options Options) int {
8288
options.NilAndErr = true
8389
s := eval.NewState()
8490
macroState := object.NewMacroEnvironment()
8591

86-
scanner := bufio.NewScanner(in)
8792
prev := ""
88-
prompt := PROMPT
93+
94+
term, err := terminal.Open()
95+
if err != nil {
96+
return log.FErrf("Error creating readline: %v", err)
97+
}
98+
defer term.Close()
99+
term.SetPrompt(PROMPT)
100+
options.Compact = true // because terminal doesn't (yet) do well will multi-line commands.
101+
term.NewHistory(options.MaxHistory)
102+
_ = term.SetHistoryFile(options.HistoryFile)
103+
// Regular expression for "!nn" to run history command nn.
104+
historyRegex := regexp.MustCompile(`^!(\d+)$`)
89105
for {
90-
fmt.Fprint(out, prompt)
91-
scanned := scanner.Scan()
92-
if !scanned {
93-
return
106+
rd, err := term.ReadLine()
107+
if errors.Is(err, io.EOF) {
108+
log.Infof("Exit requested") // Don't say EOF as ^C comes through as EOF as well.
109+
return 0
110+
}
111+
if err != nil {
112+
return log.FErrf("Error reading line: %v", err)
113+
}
114+
log.Debugf("Read: %q", rd)
115+
l := prev + rd
116+
if historyRegex.MatchString(l) {
117+
h := term.History()
118+
slices.Reverse(h)
119+
idxStr := l[1:]
120+
idx, _ := strconv.Atoi(idxStr)
121+
if idx < 1 || idx > len(h) {
122+
log.Errf("Invalid history index %d", idx)
123+
continue
124+
}
125+
l = h[idx-1]
126+
fmt.Fprintf(term.Out, "Repeating history %d: %s\n", idx, l)
127+
term.ReplaceLatest(l)
128+
}
129+
switch {
130+
case l == "history":
131+
h := term.History()
132+
slices.Reverse(h)
133+
for i, v := range h {
134+
fmt.Fprintf(term.Out, "%02d: %s\n", i+1, v)
135+
}
136+
continue
137+
case l == "help":
138+
fmt.Fprintln(term.Out, "Type 'history' to see history, '!n' to repeat history n, 'info' for language builtins")
139+
continue
94140
}
95-
l := prev + scanner.Text()
96141
// errors are already logged and this is the only case that can get contNeeded (EOL instead of EOF mode)
97-
contNeeded, _, _ := EvalOne(s, macroState, l, out, options)
142+
contNeeded, _, formatted := EvalOne(s, macroState, l, term.Out, options)
98143
if contNeeded {
99144
prev = l + "\n"
100-
prompt = CONTINUATION
145+
term.SetPrompt(CONTINUATION)
101146
} else {
147+
if prev != "" {
148+
// In addition to raw lines, we also add the single line version to history.
149+
log.LogVf("Adding to history: %q", formatted)
150+
term.AddToHistory(formatted)
151+
}
102152
prev = ""
103-
prompt = PROMPT
153+
term.SetPrompt(PROMPT)
104154
}
105155
}
106156
}

0 commit comments

Comments
 (0)