mirror of
https://github.com/beego/bee.git
synced 2024-11-17 22:20:54 +00:00
252 lines
5.6 KiB
Go
252 lines
5.6 KiB
Go
|
package terminal
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"os"
|
||
|
"os/signal"
|
||
|
"runtime"
|
||
|
"strings"
|
||
|
|
||
|
"syscall"
|
||
|
|
||
|
"github.com/peterh/liner"
|
||
|
|
||
|
"github.com/derekparker/delve/pkg/config"
|
||
|
"github.com/derekparker/delve/service"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
historyFile string = ".dbg_history"
|
||
|
terminalBlueEscapeCode string = "\033[34m"
|
||
|
terminalResetEscapeCode string = "\033[0m"
|
||
|
)
|
||
|
|
||
|
// Term represents the terminal running dlv.
|
||
|
type Term struct {
|
||
|
client service.Client
|
||
|
conf *config.Config
|
||
|
prompt string
|
||
|
line *liner.State
|
||
|
cmds *Commands
|
||
|
dumb bool
|
||
|
stdout io.Writer
|
||
|
InitFile string
|
||
|
}
|
||
|
|
||
|
// New returns a new Term.
|
||
|
func New(client service.Client, conf *config.Config) *Term {
|
||
|
cmds := DebugCommands(client)
|
||
|
if conf != nil && conf.Aliases != nil {
|
||
|
cmds.Merge(conf.Aliases)
|
||
|
}
|
||
|
|
||
|
var w io.Writer
|
||
|
|
||
|
dumb := strings.ToLower(os.Getenv("TERM")) == "dumb"
|
||
|
if dumb {
|
||
|
w = os.Stdout
|
||
|
} else {
|
||
|
w = getColorableWriter()
|
||
|
}
|
||
|
|
||
|
return &Term{
|
||
|
client: client,
|
||
|
conf: conf,
|
||
|
prompt: "(dlv) ",
|
||
|
line: liner.NewLiner(),
|
||
|
cmds: cmds,
|
||
|
dumb: dumb,
|
||
|
stdout: w,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Close returns the terminal to its previous mode.
|
||
|
func (t *Term) Close() {
|
||
|
t.line.Close()
|
||
|
}
|
||
|
|
||
|
// Run begins running dlv in the terminal.
|
||
|
func (t *Term) Run() (int, error) {
|
||
|
defer t.Close()
|
||
|
|
||
|
// Send the debugger a halt command on SIGINT
|
||
|
ch := make(chan os.Signal)
|
||
|
signal.Notify(ch, syscall.SIGINT)
|
||
|
go func() {
|
||
|
for range ch {
|
||
|
fmt.Printf("received SIGINT, stopping process (will not forward signal)")
|
||
|
_, err := t.client.Halt()
|
||
|
if err != nil {
|
||
|
fmt.Fprintf(os.Stderr, "%v", err)
|
||
|
}
|
||
|
}
|
||
|
}()
|
||
|
|
||
|
t.line.SetCompleter(func(line string) (c []string) {
|
||
|
for _, cmd := range t.cmds.cmds {
|
||
|
for _, alias := range cmd.aliases {
|
||
|
if strings.HasPrefix(alias, strings.ToLower(line)) {
|
||
|
c = append(c, alias)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return
|
||
|
})
|
||
|
|
||
|
fullHistoryFile, err := config.GetConfigFilePath(historyFile)
|
||
|
if err != nil {
|
||
|
fmt.Printf("Unable to load history file: %v.", err)
|
||
|
}
|
||
|
|
||
|
f, err := os.Open(fullHistoryFile)
|
||
|
if err != nil {
|
||
|
f, err = os.Create(fullHistoryFile)
|
||
|
if err != nil {
|
||
|
fmt.Printf("Unable to open history file: %v. History will not be saved for this session.", err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
t.line.ReadHistory(f)
|
||
|
f.Close()
|
||
|
fmt.Println("Type 'help' for list of commands.")
|
||
|
|
||
|
if t.InitFile != "" {
|
||
|
err := t.cmds.executeFile(t, t.InitFile)
|
||
|
if err != nil {
|
||
|
fmt.Fprintf(os.Stderr, "Error executing init file: %s\n", err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for {
|
||
|
cmdstr, err := t.promptForInput()
|
||
|
if err != nil {
|
||
|
if err == io.EOF {
|
||
|
fmt.Println("exit")
|
||
|
return t.handleExit()
|
||
|
}
|
||
|
return 1, fmt.Errorf("Prompt for input failed.\n")
|
||
|
}
|
||
|
|
||
|
cmdstr, args := parseCommand(cmdstr)
|
||
|
if err := t.cmds.Call(cmdstr, args, t); err != nil {
|
||
|
if _, ok := err.(ExitRequestError); ok {
|
||
|
return t.handleExit()
|
||
|
}
|
||
|
// The type information gets lost in serialization / de-serialization,
|
||
|
// so we do a string compare on the error message to see if the process
|
||
|
// has exited, or if the command actually failed.
|
||
|
if strings.Contains(err.Error(), "exited") {
|
||
|
fmt.Fprintln(os.Stderr, err.Error())
|
||
|
} else {
|
||
|
fmt.Fprintf(os.Stderr, "Command failed: %s\n", err)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Println prints a line to the terminal.
|
||
|
func (t *Term) Println(prefix, str string) {
|
||
|
if !t.dumb {
|
||
|
prefix = fmt.Sprintf("%s%s%s", terminalBlueEscapeCode, prefix, terminalResetEscapeCode)
|
||
|
}
|
||
|
fmt.Fprintf(t.stdout, "%s%s\n", prefix, str)
|
||
|
}
|
||
|
|
||
|
// Substitues directory to source file.
|
||
|
//
|
||
|
// Ensures that only directory is substitued, for example:
|
||
|
// substitute from `/dir/subdir`, substitute to `/new`
|
||
|
// for file path `/dir/subdir/file` will return file path `/new/file`.
|
||
|
// for file path `/dir/subdir-2/file` substitution will not be applied.
|
||
|
//
|
||
|
// If more than one substitution rule is defined, the rules are applied
|
||
|
// in the order they are defined, first rule that matches is used for
|
||
|
// substitution.
|
||
|
func (t *Term) substitutePath(path string) string {
|
||
|
path = crossPlatformPath(path)
|
||
|
if t.conf == nil {
|
||
|
return path
|
||
|
}
|
||
|
separator := string(os.PathSeparator)
|
||
|
for _, r := range t.conf.SubstitutePath {
|
||
|
from := crossPlatformPath(r.From)
|
||
|
to := r.To
|
||
|
|
||
|
if !strings.HasSuffix(from, separator) {
|
||
|
from = from + separator
|
||
|
}
|
||
|
if !strings.HasSuffix(to, separator) {
|
||
|
to = to + separator
|
||
|
}
|
||
|
if strings.HasPrefix(path, from) {
|
||
|
return strings.Replace(path, from, to, 1)
|
||
|
}
|
||
|
}
|
||
|
return path
|
||
|
}
|
||
|
|
||
|
func crossPlatformPath(path string) string {
|
||
|
if runtime.GOOS == "darwin" || runtime.GOOS == "windows" {
|
||
|
return strings.ToLower(path)
|
||
|
}
|
||
|
return path
|
||
|
}
|
||
|
|
||
|
func (t *Term) promptForInput() (string, error) {
|
||
|
l, err := t.line.Prompt(t.prompt)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
|
||
|
l = strings.TrimSuffix(l, "\n")
|
||
|
if l != "" {
|
||
|
t.line.AppendHistory(l)
|
||
|
}
|
||
|
|
||
|
return l, nil
|
||
|
}
|
||
|
|
||
|
func (t *Term) handleExit() (int, error) {
|
||
|
fullHistoryFile, err := config.GetConfigFilePath(historyFile)
|
||
|
if err != nil {
|
||
|
fmt.Println("Error saving history file:", err)
|
||
|
} else {
|
||
|
if f, err := os.OpenFile(fullHistoryFile, os.O_RDWR, 0666); err == nil {
|
||
|
_, err = t.line.WriteHistory(f)
|
||
|
if err != nil {
|
||
|
fmt.Println("readline history error:", err)
|
||
|
}
|
||
|
f.Close()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
s, err := t.client.GetState()
|
||
|
if err != nil {
|
||
|
return 1, err
|
||
|
}
|
||
|
if !s.Exited {
|
||
|
kill := true
|
||
|
if t.client.AttachedToExistingProcess() {
|
||
|
answer, err := t.line.Prompt("Would you like to kill the process? [Y/n] ")
|
||
|
if err != nil {
|
||
|
return 2, io.EOF
|
||
|
}
|
||
|
answer = strings.ToLower(strings.TrimSpace(answer))
|
||
|
kill = (answer != "n" && answer != "no")
|
||
|
}
|
||
|
if err := t.client.Detach(kill); err != nil {
|
||
|
return 1, err
|
||
|
}
|
||
|
}
|
||
|
return 0, nil
|
||
|
}
|
||
|
|
||
|
func parseCommand(cmdstr string) (string, string) {
|
||
|
vals := strings.SplitN(cmdstr, " ", 2)
|
||
|
if len(vals) == 1 {
|
||
|
return vals[0], ""
|
||
|
}
|
||
|
return vals[0], strings.TrimSpace(vals[1])
|
||
|
}
|