2022-02-24 04:29:48 +02:00
|
|
|
package custom_commands
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"errors"
|
|
|
|
"regexp"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"text/template"
|
|
|
|
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/common"
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/style"
|
|
|
|
)
|
|
|
|
|
|
|
|
type MenuGenerator struct {
|
|
|
|
c *common.Common
|
|
|
|
}
|
|
|
|
|
|
|
|
// takes the output of a command and returns a list of menu entries based on a filter
|
|
|
|
// and value/label format templates provided by the user
|
|
|
|
func NewMenuGenerator(c *common.Common) *MenuGenerator {
|
|
|
|
return &MenuGenerator{c: c}
|
|
|
|
}
|
|
|
|
|
2023-05-29 14:52:16 +02:00
|
|
|
type commandMenuItem struct {
|
2022-02-24 04:29:48 +02:00
|
|
|
label string
|
|
|
|
value string
|
|
|
|
}
|
|
|
|
|
2023-05-29 14:52:16 +02:00
|
|
|
func (self *MenuGenerator) call(commandOutput, filter, valueFormat, labelFormat string) ([]*commandMenuItem, error) {
|
|
|
|
menuItemFromLine, err := self.getMenuItemFromLinefn(filter, valueFormat, labelFormat)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
menuItems := []*commandMenuItem{}
|
|
|
|
for _, line := range strings.Split(commandOutput, "\n") {
|
|
|
|
if line == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
menuItem, err := menuItemFromLine(line)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
menuItems = append(menuItems, menuItem)
|
|
|
|
}
|
|
|
|
|
|
|
|
return menuItems, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MenuGenerator) getMenuItemFromLinefn(filter string, valueFormat string, labelFormat string) (func(line string) (*commandMenuItem, error), error) {
|
|
|
|
if filter == "" && valueFormat == "" && labelFormat == "" {
|
|
|
|
// showing command output lines as-is in suggestions panel
|
|
|
|
return func(line string) (*commandMenuItem, error) {
|
|
|
|
return &commandMenuItem{label: line, value: line}, nil
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2022-02-24 04:29:48 +02:00
|
|
|
regex, err := regexp.Compile(filter)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("unable to parse filter regex, error: " + err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
valueTemplateAux, err := template.New("format").Parse(valueFormat)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("unable to parse value format, error: " + err.Error())
|
|
|
|
}
|
|
|
|
valueTemplate := NewTrimmerTemplate(valueTemplateAux)
|
|
|
|
|
|
|
|
var labelTemplate *TrimmerTemplate
|
|
|
|
if labelFormat != "" {
|
|
|
|
colorFuncMap := style.TemplateFuncMapAddColors(template.FuncMap{})
|
|
|
|
labelTemplateAux, err := template.New("format").Funcs(colorFuncMap).Parse(labelFormat)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("unable to parse label format, error: " + err.Error())
|
|
|
|
}
|
|
|
|
labelTemplate = NewTrimmerTemplate(labelTemplateAux)
|
|
|
|
} else {
|
|
|
|
labelTemplate = valueTemplate
|
|
|
|
}
|
|
|
|
|
2023-05-29 14:52:16 +02:00
|
|
|
return func(line string) (*commandMenuItem, error) {
|
|
|
|
return self.generateMenuItem(
|
2022-02-24 04:29:48 +02:00
|
|
|
line,
|
|
|
|
regex,
|
|
|
|
valueTemplate,
|
|
|
|
labelTemplate,
|
|
|
|
)
|
2023-05-29 14:52:16 +02:00
|
|
|
}, nil
|
2022-02-24 04:29:48 +02:00
|
|
|
}
|
|
|
|
|
2023-05-29 14:52:16 +02:00
|
|
|
func (self *MenuGenerator) generateMenuItem(
|
2022-02-24 04:29:48 +02:00
|
|
|
line string,
|
|
|
|
regex *regexp.Regexp,
|
|
|
|
valueTemplate *TrimmerTemplate,
|
|
|
|
labelTemplate *TrimmerTemplate,
|
2023-05-29 14:52:16 +02:00
|
|
|
) (*commandMenuItem, error) {
|
2022-02-24 04:29:48 +02:00
|
|
|
tmplData := self.parseLine(line, regex)
|
|
|
|
|
2023-05-29 14:52:16 +02:00
|
|
|
entry := &commandMenuItem{}
|
2022-02-24 04:29:48 +02:00
|
|
|
|
|
|
|
var err error
|
|
|
|
entry.value, err = valueTemplate.execute(tmplData)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
entry.label, err = labelTemplate.execute(tmplData)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return entry, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *MenuGenerator) parseLine(line string, regex *regexp.Regexp) map[string]string {
|
|
|
|
tmplData := map[string]string{}
|
|
|
|
out := regex.FindAllStringSubmatch(line, -1)
|
|
|
|
if len(out) > 0 {
|
|
|
|
for groupIdx, group := range regex.SubexpNames() {
|
|
|
|
// Record matched group with group ids
|
|
|
|
matchName := "group_" + strconv.Itoa(groupIdx)
|
|
|
|
tmplData[matchName] = out[0][groupIdx]
|
|
|
|
// Record last named group non-empty matches as group matches
|
|
|
|
if group != "" {
|
|
|
|
tmplData[group] = out[0][groupIdx]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return tmplData
|
|
|
|
}
|
|
|
|
|
|
|
|
// wrapper around a template which trims the output
|
|
|
|
type TrimmerTemplate struct {
|
|
|
|
template *template.Template
|
|
|
|
buffer *bytes.Buffer
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewTrimmerTemplate(template *template.Template) *TrimmerTemplate {
|
|
|
|
return &TrimmerTemplate{
|
|
|
|
template: template,
|
|
|
|
buffer: bytes.NewBuffer(nil),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *TrimmerTemplate) execute(tmplData map[string]string) (string, error) {
|
|
|
|
self.buffer.Reset()
|
|
|
|
err := self.template.Execute(self.buffer, tmplData)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return strings.TrimSpace(self.buffer.String()), nil
|
|
|
|
}
|