mirror of
				https://github.com/jesseduffield/lazygit.git
				synced 2025-10-25 05:37:37 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			257 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			257 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package commands
 | |
| 
 | |
| import (
 | |
| 	"bufio"
 | |
| 	"errors"
 | |
| 	"os"
 | |
| 	"os/exec"
 | |
| 	"regexp"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/ionrock/procs"
 | |
| 	"github.com/jesseduffield/lazygit/pkg/config"
 | |
| 	"github.com/jesseduffield/lazygit/pkg/utils"
 | |
| 	"github.com/kr/pty"
 | |
| 
 | |
| 	"github.com/mgutz/str"
 | |
| 
 | |
| 	"github.com/sirupsen/logrus"
 | |
| 	gitconfig "github.com/tcnksm/go-gitconfig"
 | |
| )
 | |
| 
 | |
| // Platform stores the os state
 | |
| type Platform struct {
 | |
| 	os                   string
 | |
| 	shell                string
 | |
| 	shellArg             string
 | |
| 	escapedQuote         string
 | |
| 	openCommand          string
 | |
| 	fallbackEscapedQuote string
 | |
| }
 | |
| 
 | |
| // OSCommand holds all the os commands
 | |
| type OSCommand struct {
 | |
| 	Log                *logrus.Entry
 | |
| 	Platform           *Platform
 | |
| 	Config             config.AppConfigurer
 | |
| 	command            func(string, ...string) *exec.Cmd
 | |
| 	getGlobalGitConfig func(string) (string, error)
 | |
| 	getenv             func(string) string
 | |
| }
 | |
| 
 | |
| // NewOSCommand os command runner
 | |
| func NewOSCommand(log *logrus.Entry, config config.AppConfigurer) *OSCommand {
 | |
| 	return &OSCommand{
 | |
| 		Log:                log,
 | |
| 		Platform:           getPlatform(),
 | |
| 		Config:             config,
 | |
| 		command:            exec.Command,
 | |
| 		getGlobalGitConfig: gitconfig.Global,
 | |
| 		getenv:             os.Getenv,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // RunCommandWithOutput wrapper around commands returning their output and error
 | |
| func (c *OSCommand) RunCommandWithOutput(command string) (string, error) {
 | |
| 	c.Log.WithField("command", command).Info("RunCommand")
 | |
| 	splitCmd := str.ToArgv(command)
 | |
| 	c.Log.Info(splitCmd)
 | |
| 	return sanitisedCommandOutput(
 | |
| 		c.command(splitCmd[0], splitCmd[1:]...).CombinedOutput(),
 | |
| 	)
 | |
| }
 | |
| 
 | |
| // RunCommandWithOutputLive runs a command and return every word that gets written in stdout
 | |
| // Output is a function that executes by every word that gets read by bufio
 | |
| // As return of output you need to give a string that will be written to stdin
 | |
| // NOTE: If the return data is empty it won't written anything to stdin
 | |
| // NOTE: You don't have to include a enter in the return data this function will do that for you
 | |
| func (c *OSCommand) RunCommandWithOutputLive(command string, output func(string) string) error {
 | |
| 	splitCmd := str.ToArgv(command)
 | |
| 	cmd := exec.Command(splitCmd[0], splitCmd[1:]...)
 | |
| 
 | |
| 	cmd.Env = procs.Env(map[string]string{
 | |
| 		"LANG":   "en_US.utf8",
 | |
| 		"LC_ALL": "en_US.UTF-8",
 | |
| 	}, true)
 | |
| 
 | |
| 	tty, err := pty.Start(cmd)
 | |
| 
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	defer func() { _ = tty.Close() }()
 | |
| 
 | |
| 	go func() {
 | |
| 		// Regex to cleanup the command output
 | |
| 		// sometimes the output words include unneeded spaces at eatch end of the string
 | |
| 		re := regexp.MustCompile(`(^\s*)|(\s*$)`)
 | |
| 
 | |
| 		scanner := bufio.NewScanner(tty)
 | |
| 		scanner.Split(bufio.ScanWords)
 | |
| 		for scanner.Scan() {
 | |
| 			toWrite := output(re.ReplaceAllString(scanner.Text(), ""))
 | |
| 			if len(toWrite) > 0 {
 | |
| 				_, _ = tty.Write([]byte(toWrite + "\n"))
 | |
| 			}
 | |
| 		}
 | |
| 	}()
 | |
| 
 | |
| 	if err := cmd.Wait(); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // DetectUnamePass detect a username / password question in a command
 | |
| // ask is a function that gets executen when this function detect you need to fillin a password
 | |
| // The ask argument will be "username" or "password" and expects the user's password or username back
 | |
| func (c *OSCommand) DetectUnamePass(command string, ask func(string) string) error {
 | |
| 	ttyText := ""
 | |
| 	errors := []error{}
 | |
| 	err := c.RunCommandWithOutputLive(command, func(word string) string {
 | |
| 		ttyText = ttyText + " " + word
 | |
| 
 | |
| 		// detect username question
 | |
| 		detectUname, err := regexp.MatchString(`Username\s*for\s*'.+':`, ttyText)
 | |
| 		if err != nil {
 | |
| 			errors = append(errors, err)
 | |
| 		}
 | |
| 		if detectUname {
 | |
| 			// reset the text and return the user's username
 | |
| 			ttyText = ""
 | |
| 			return ask("username")
 | |
| 		}
 | |
| 
 | |
| 		// detect password question
 | |
| 		detectPass, err := regexp.MatchString(`Password\s*for\s*'.+':`, ttyText)
 | |
| 		if err != nil {
 | |
| 			errors = append(errors, err)
 | |
| 		}
 | |
| 		if detectPass {
 | |
| 			// reset the text and return the user's username
 | |
| 			ttyText = ""
 | |
| 			return ask("password")
 | |
| 		}
 | |
| 		return ""
 | |
| 	})
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	if len(errors) > 0 {
 | |
| 		return errors[0]
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // RunCommand runs a command and just returns the error
 | |
| func (c *OSCommand) RunCommand(command string) error {
 | |
| 	_, err := c.RunCommandWithOutput(command)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| // FileType tells us if the file is a file, directory or other
 | |
| func (c *OSCommand) FileType(path string) string {
 | |
| 	fileInfo, err := os.Stat(path)
 | |
| 	if err != nil {
 | |
| 		return "other"
 | |
| 	}
 | |
| 	if fileInfo.IsDir() {
 | |
| 		return "directory"
 | |
| 	}
 | |
| 	return "file"
 | |
| }
 | |
| 
 | |
| // RunDirectCommand wrapper around direct commands
 | |
| func (c *OSCommand) RunDirectCommand(command string) (string, error) {
 | |
| 	c.Log.WithField("command", command).Info("RunDirectCommand")
 | |
| 
 | |
| 	return sanitisedCommandOutput(
 | |
| 		c.command(c.Platform.shell, c.Platform.shellArg, command).
 | |
| 			CombinedOutput(),
 | |
| 	)
 | |
| }
 | |
| 
 | |
| func sanitisedCommandOutput(output []byte, err error) (string, error) {
 | |
| 	outputString := string(output)
 | |
| 	if err != nil {
 | |
| 		// errors like 'exit status 1' are not very useful so we'll create an error
 | |
| 		// from the combined output
 | |
| 		if outputString == "" {
 | |
| 			return "", err
 | |
| 		}
 | |
| 		return outputString, errors.New(outputString)
 | |
| 	}
 | |
| 	return outputString, nil
 | |
| }
 | |
| 
 | |
| // OpenFile opens a file with the given
 | |
| func (c *OSCommand) OpenFile(filename string) error {
 | |
| 	commandTemplate := c.Config.GetUserConfig().GetString("os.openCommand")
 | |
| 	templateValues := map[string]string{
 | |
| 		"filename": c.Quote(filename),
 | |
| 	}
 | |
| 
 | |
| 	command := utils.ResolvePlaceholderString(commandTemplate, templateValues)
 | |
| 	err := c.RunCommand(command)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| // EditFile opens a file in a subprocess using whatever editor is available,
 | |
| // falling back to core.editor, VISUAL, EDITOR, then vi
 | |
| func (c *OSCommand) EditFile(filename string) (*exec.Cmd, error) {
 | |
| 	editor, _ := c.getGlobalGitConfig("core.editor")
 | |
| 
 | |
| 	if editor == "" {
 | |
| 		editor = c.getenv("VISUAL")
 | |
| 	}
 | |
| 	if editor == "" {
 | |
| 		editor = c.getenv("EDITOR")
 | |
| 	}
 | |
| 	if editor == "" {
 | |
| 		if err := c.RunCommand("which vi"); err == nil {
 | |
| 			editor = "vi"
 | |
| 		}
 | |
| 	}
 | |
| 	if editor == "" {
 | |
| 		return nil, errors.New("No editor defined in $VISUAL, $EDITOR, or git config")
 | |
| 	}
 | |
| 
 | |
| 	return c.PrepareSubProcess(editor, filename), nil
 | |
| }
 | |
| 
 | |
| // PrepareSubProcess iniPrepareSubProcessrocess then tells the Gui to switch to it
 | |
| func (c *OSCommand) PrepareSubProcess(cmdName string, commandArgs ...string) *exec.Cmd {
 | |
| 	return c.command(cmdName, commandArgs...)
 | |
| }
 | |
| 
 | |
| // Quote wraps a message in platform-specific quotation marks
 | |
| func (c *OSCommand) Quote(message string) string {
 | |
| 	message = strings.Replace(message, "`", "\\`", -1)
 | |
| 	escapedQuote := c.Platform.escapedQuote
 | |
| 	if strings.Contains(message, c.Platform.escapedQuote) {
 | |
| 		escapedQuote = c.Platform.fallbackEscapedQuote
 | |
| 	}
 | |
| 	return escapedQuote + message + escapedQuote
 | |
| }
 | |
| 
 | |
| // Unquote removes wrapping quotations marks if they are present
 | |
| // this is needed for removing quotes from staged filenames with spaces
 | |
| func (c *OSCommand) Unquote(message string) string {
 | |
| 	return strings.Replace(message, `"`, "", -1)
 | |
| }
 | |
| 
 | |
| // AppendLineToFile adds a new line in file
 | |
| func (c *OSCommand) AppendLineToFile(filename, line string) error {
 | |
| 	f, err := os.OpenFile(filename, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	defer f.Close()
 | |
| 
 | |
| 	_, err = f.WriteString("\n" + line)
 | |
| 	return err
 | |
| }
 |