buffer for terminal input/output --> buf + term + io
Buftermio is an importable Go package. It provides a cached buffer for input from the CLI. Perfect for a REPL!
Basically, an improved interface for a bufio.Scanner reading os.Stdin.
When you use a CLI shell, you may take for granted that you have a cached history of your commands. Then, when you go to use your own CLI program, suddenly you've lost that cache and have to type each command anew instead of just pressing up to scroll through previous commands. Here is buftermio to the rescue. Buftermio lets you scroll through your previous inputs with the up and down arrow keys!
Inspired by the bytes package Buffer struct. Buftermio uses a bufio.Scanner to read from os.Stdin for input. It temporarily prevents the CLI shell from printing directly to Stdout while buftermio reads from Stdin. Buftermio intercepts each key to interpret them before printing them to Stdout.
Run
go get github.com/skovranek/buftermio
I made this to be as simple and straight forward as possible. The only configuration is an optional string prompt like a shell would have.
You have two ways to set the prompt. First, you can set a default prompt with the NewBuffer function. Second, you can set the prompt case-by-case whenever you call the GetInput method. Both NewBuffer and GetInput take variatic string arguments which form the prompt which prints out before getting input.
See the next section, "Implement", for details.
- Instantiate the buffer once with the NewBuffer function from buftermio. It takes an optional variatic string argument which is joined to form the prompt which prints out before getting input.
buffer := buftermio.NewBuffer("Hello ", username, ": ")
// or
buffer := buftermio.NewBuffer("$ ")
// or
buffer := buftermio.NewBuffer()
- Call the GetInput method on the buffer each time you want to get input from the CLI. It takes a optional variatic string argument that overrides the default prompt. GetInput returns a string.
Input, err := buffer.GetInput("Input from ", username, ": ")
// or
Input, err := buffer.GetInput(fmt.Sprintf("Query #%v: ", i))
// or
Input, err := buffer.GetInput()
Here is a simple REPL that uses buftermio. Each loop it prints the prompt "input: " and waits for input from the terminal through buftermio, then echos the input and prints the keycodes. Potentially useful for checking keycodes.
package main
import (
"fmt"
"github.com/skovranek/buftermio"
)
func main() {
buffer := buftermio.NewBuffer("input: ")
for {
input, err := buffer.GetInput()
if err != nil {
fmt.Println(err)
continue
}
if input == "q" {
return
}
fmt.Printf("output: %s\nbytes: %v\n\n", input, []byte(input))
}
}
Using buftermio should feel like using the interface of a shell like bash or zsh.
- Up/down arrows to scroll though the cached history of inputs. The cursor stays in place for each input.
- Left/right arrows to move the cursor left/right. Buftermio will insert characters instead of overwriting them.
- Return to enter input.
- Tab will output four spaces. I found the tab functionality was inconsistent in the CLI, so I simplified it.
- Ctrl+A will move the cursor to the start of the line.
- Ctrl+E will move the cursor to the end of the line.
- Ctrl+W will delete the previous word.
- Ctrl+U will delete from the cursor to the start of the line.
- Ctrl+K will delete from the cursor to the end of the line.
WARNING Pressing keys or key combinations that produce keycode sequences, such as <Alt+A>, may not work as expected. This is a known issue and may cause errors because buftermio is not a keylogger counting keystrokes. It is only counting the character bytes from Stdin. Letters, numbers and most special characters will behave normally. But, if on the off chance there are unpredicted effects when you use certain keys, you now know why. I'll gladly add checks for certain sequences for anyone that requires it.
Buftermio uses a subrepository from the Go Project called execabs instead of the standard library's "os/exec" package because of a path-security issue with "os/exec". No other dependencies are included.
Manually tested with a zsh shell in the macOS terminal. Mocking stdin would only test the mock. Please let me know if buftermio does not work as expected in your environment.
Questions, issues or suggestions: mattjskov at gmail.com
Feel free to chip in. Let's work together to customize buftermio for your project. Submit pull requests to the 'main' branch.