1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-01-18 05:17:55 +02:00
lazygit/pkg/gui/custom_commands.go

306 lines
9.5 KiB
Go
Raw Normal View History

2020-09-26 15:23:28 +10:00
package gui
import (
"bytes"
"errors"
2020-09-26 17:15:13 +10:00
"log"
"regexp"
"strconv"
2020-09-27 09:37:22 +10:00
"strings"
"text/template"
2020-09-26 15:23:28 +10:00
2020-09-27 09:13:31 +10:00
"github.com/fatih/color"
2020-09-26 17:15:13 +10:00
"github.com/jesseduffield/gocui"
"github.com/jesseduffield/lazygit/pkg/commands/models"
2020-10-03 14:54:55 +10:00
"github.com/jesseduffield/lazygit/pkg/config"
2020-09-27 09:13:31 +10:00
"github.com/jesseduffield/lazygit/pkg/utils"
2020-09-26 15:23:28 +10:00
)
type CustomCommandObjects struct {
2021-03-31 22:08:55 +11:00
SelectedLocalCommit *models.Commit
SelectedReflogCommit *models.Commit
SelectedSubCommit *models.Commit
SelectedFile *models.File
SelectedPath string
SelectedLocalBranch *models.Branch
SelectedRemoteBranch *models.RemoteBranch
SelectedRemote *models.Remote
SelectedTag *models.Tag
SelectedStashEntry *models.StashEntry
SelectedCommitFile *models.CommitFile
SelectedCommitFilePath string
CheckedOutBranch *models.Branch
PromptResponses []string
2020-09-26 19:48:13 +10:00
}
func (gui *Gui) resolveTemplate(templateStr string, promptResponses []string) (string, error) {
objects := CustomCommandObjects{
2021-03-31 22:08:55 +11:00
SelectedFile: gui.getSelectedFile(),
SelectedPath: gui.getSelectedPath(),
SelectedLocalCommit: gui.getSelectedLocalCommit(),
SelectedReflogCommit: gui.getSelectedReflogCommit(),
SelectedLocalBranch: gui.getSelectedBranch(),
SelectedRemoteBranch: gui.getSelectedRemoteBranch(),
SelectedRemote: gui.getSelectedRemote(),
SelectedTag: gui.getSelectedTag(),
SelectedStashEntry: gui.getSelectedStashEntry(),
SelectedCommitFile: gui.getSelectedCommitFile(),
SelectedCommitFilePath: gui.getSelectedCommitFilePath(),
SelectedSubCommit: gui.getSelectedSubCommit(),
CheckedOutBranch: gui.currentBranch(),
PromptResponses: promptResponses,
2020-09-26 19:48:13 +10:00
}
2020-09-30 21:12:03 +10:00
return utils.ResolveTemplate(templateStr, objects)
2020-09-26 15:23:28 +10:00
}
func (gui *Gui) inputPrompt(prompt config.CustomCommandPrompt, promptResponses []string, responseIdx int, wrappedF func() error) error {
title, err := gui.resolveTemplate(prompt.Title, promptResponses)
if err != nil {
return gui.surfaceError(err)
}
initialValue, err := gui.resolveTemplate(prompt.InitialValue, promptResponses)
if err != nil {
return gui.surfaceError(err)
}
return gui.prompt(promptOpts{
title: title,
initialContent: initialValue,
handleConfirm: func(str string) error {
promptResponses[responseIdx] = str
return wrappedF()
},
})
}
func (gui *Gui) menuPrompt(prompt config.CustomCommandPrompt, promptResponses []string, responseIdx int, wrappedF func() error) error {
// need to make a menu here some how
menuItems := make([]*menuItem, len(prompt.Options))
for i, option := range prompt.Options {
option := option
nameTemplate := option.Name
if nameTemplate == "" {
// this allows you to only pass values rather than bother with names/descriptions
nameTemplate = option.Value
}
name, err := gui.resolveTemplate(nameTemplate, promptResponses)
if err != nil {
return gui.surfaceError(err)
}
description, err := gui.resolveTemplate(option.Description, promptResponses)
if err != nil {
return gui.surfaceError(err)
}
value, err := gui.resolveTemplate(option.Value, promptResponses)
if err != nil {
return gui.surfaceError(err)
}
menuItems[i] = &menuItem{
displayStrings: []string{name, utils.ColoredString(description, color.FgYellow)},
onPress: func() error {
promptResponses[responseIdx] = value
return wrappedF()
},
}
}
title, err := gui.resolveTemplate(prompt.Title, promptResponses)
if err != nil {
return gui.surfaceError(err)
}
return gui.createMenu(title, menuItems, createMenuOptions{showCancel: true})
}
func (gui *Gui) generateMenuCandidates(commandOutput string, filter string, format string) ([]string, error) {
candidates := []string{}
reg, err := regexp.Compile(filter)
if err != nil {
return candidates, gui.surfaceError(errors.New("unable to parse filter regex, error: " + err.Error()))
}
buff := bytes.NewBuffer(nil)
temp, err := template.New("format").Parse(format)
if err != nil {
return candidates, gui.surfaceError(errors.New("unable to parse format, error: " + err.Error()))
}
for _, str := range strings.Split(string(commandOutput), "\n") {
if str == "" {
continue
}
tmplData := map[string]string{}
out := reg.FindAllStringSubmatch(str, -1)
if len(out) > 0 {
for groupIdx, group := range reg.SubexpNames() {
// Record matched group with group ids
matchName := "group_" + strconv.Itoa(groupIdx)
tmplData[matchName] = group
// Record last named group non-empty matches as group matches
if group != "" {
tmplData[group] = out[0][groupIdx]
}
}
}
err = temp.Execute(buff, tmplData)
if err != nil {
return candidates, gui.surfaceError(err)
}
candidates = append(candidates, strings.TrimSpace(buff.String()))
buff.Reset()
}
return candidates, err
}
func (gui *Gui) menuPromptFromCommand(prompt config.CustomCommandPrompt, promptResponses []string, responseIdx int, wrappedF func() error) error {
// Collect cmd to run from config
cmdStr, err := gui.resolveTemplate(prompt.Command, promptResponses)
if err != nil {
return gui.surfaceError(err)
}
// Collect Filter regexp
filter, err := gui.resolveTemplate(prompt.Filter, promptResponses)
if err != nil {
return gui.surfaceError(err)
}
// Run and save output
message, err := gui.GitCommand.RunCommandWithOutput(cmdStr)
if err != nil {
return gui.surfaceError(err)
}
// Need to make a menu out of what the cmd has displayed
candidates, err := gui.generateMenuCandidates(message, filter, prompt.Format)
menuItems := make([]*menuItem, len(candidates))
for i := range candidates {
menuItems[i] = &menuItem{
displayStrings: []string{candidates[i]},
onPress: func() error {
promptResponses[responseIdx] = candidates[i]
return wrappedF()
},
}
}
title, err := gui.resolveTemplate(prompt.Title, promptResponses)
if err != nil {
return gui.surfaceError(err)
}
return gui.createMenu(title, menuItems, createMenuOptions{showCancel: true})
}
2020-10-03 14:54:55 +10:00
func (gui *Gui) handleCustomCommandKeybinding(customCommand config.CustomCommand) func() error {
2020-09-26 15:23:28 +10:00
return func() error {
2020-09-26 19:48:13 +10:00
promptResponses := make([]string, len(customCommand.Prompts))
2020-09-26 15:23:28 +10:00
2020-09-26 19:48:13 +10:00
f := func() error {
cmdStr, err := gui.resolveTemplate(customCommand.Command, promptResponses)
if err != nil {
return gui.surfaceError(err)
}
2020-09-26 15:23:28 +10:00
2020-09-26 19:48:13 +10:00
if customCommand.Subprocess {
2021-04-10 11:40:42 +10:00
return gui.runSubprocessWithSuspenseAndRefresh(gui.OSCommand.PrepareShellSubProcess(cmdStr))
2020-09-26 19:48:13 +10:00
}
2020-09-26 15:23:28 +10:00
2020-09-27 09:21:20 +10:00
loadingText := customCommand.LoadingText
if loadingText == "" {
2020-10-04 11:00:48 +11:00
loadingText = gui.Tr.LcRunningCustomCommandStatus
2020-09-27 09:21:20 +10:00
}
return gui.WithWaitingStatus(loadingText, func() error {
2021-04-11 19:35:42 +10:00
if err := gui.OSCommand.WithSpan(gui.Tr.Spans.CustomCommand).RunShellCommand(cmdStr); err != nil {
2020-09-26 19:48:13 +10:00
return gui.surfaceError(err)
}
return gui.refreshSidePanels(refreshOptions{})
})
2020-09-26 17:15:13 +10:00
}
2020-09-26 19:48:13 +10:00
// if we have prompts we'll recursively wrap our confirm handlers with more prompts
// until we reach the actual command
for reverseIdx := range customCommand.Prompts {
idx := len(customCommand.Prompts) - 1 - reverseIdx
2020-09-26 17:15:13 +10:00
2020-09-26 19:48:13 +10:00
// going backwards so the outermost prompt is the first one
prompt := customCommand.Prompts[idx]
// need to do this because f's value will change with each iteration
wrappedF := f
2020-09-26 20:32:19 +10:00
switch prompt.Type {
2020-09-27 09:13:31 +10:00
case "input":
2020-09-26 20:32:19 +10:00
f = func() error {
return gui.inputPrompt(prompt, promptResponses, idx, wrappedF)
2020-09-26 20:32:19 +10:00
}
case "menu":
f = func() error {
return gui.menuPrompt(prompt, promptResponses, idx, wrappedF)
2021-07-17 18:02:11 +01:00
}
case "menuFromCommand":
f = func() error {
return gui.menuPromptFromCommand(prompt, promptResponses, idx, wrappedF)
2020-09-26 20:32:19 +10:00
}
default:
2021-07-17 18:02:11 +01:00
return gui.createErrorPanel("custom command prompt must have a type of 'input', 'menu' or 'menuFromCommand'")
2020-09-26 15:23:28 +10:00
}
2020-09-26 20:32:19 +10:00
2020-09-26 19:48:13 +10:00
}
return f()
2020-09-26 15:23:28 +10:00
}
}
2020-09-26 17:15:13 +10:00
func (gui *Gui) GetCustomCommandKeybindings() []*Binding {
bindings := []*Binding{}
2020-10-03 14:54:55 +10:00
customCommands := gui.Config.GetUserConfig().CustomCommands
2020-09-26 17:15:13 +10:00
for _, customCommand := range customCommands {
var viewName string
var contexts []string
2020-09-27 09:37:22 +10:00
switch customCommand.Context {
case "global":
2020-09-26 17:15:13 +10:00
viewName = ""
2020-09-27 09:37:22 +10:00
case "":
log.Fatalf("Error parsing custom command keybindings: context not provided (use context: 'global' for the global context). Key: %s, Command: %s", customCommand.Key, customCommand.Command)
default:
2021-04-04 23:51:59 +10:00
context, ok := gui.contextForContextKey(ContextKey(customCommand.Context))
// stupid golang making me build an array of strings for this.
allContextKeyStrings := make([]string, len(allContextKeys))
for i := range allContextKeys {
allContextKeyStrings[i] = string(allContextKeys[i])
}
2020-09-27 09:37:22 +10:00
if !ok {
2021-04-04 23:51:59 +10:00
log.Fatalf("Error when setting custom command keybindings: unknown context: %s. Key: %s, Command: %s.\nPermitted contexts: %s", customCommand.Context, customCommand.Key, customCommand.Command, strings.Join(allContextKeyStrings, ", "))
2020-09-26 17:15:13 +10:00
}
// here we assume that a given context will always belong to the same view.
// Currently this is a safe bet but it's by no means guaranteed in the long term
// and we might need to make some changes in the future to support it.
viewName = context.GetViewName()
contexts = []string{customCommand.Context}
2020-09-26 17:15:13 +10:00
}
description := customCommand.Description
if description == "" {
description = customCommand.Command
}
2020-09-26 17:15:13 +10:00
bindings = append(bindings, &Binding{
ViewName: viewName,
Contexts: contexts,
2020-09-26 17:15:13 +10:00
Key: gui.getKey(customCommand.Key),
Modifier: gocui.ModNone,
Handler: gui.handleCustomCommandKeybinding(customCommand),
Description: description,
2020-09-26 17:15:13 +10:00
})
}
return bindings
}