1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-01-22 05:29:44 +02:00

458 lines
13 KiB
Go
Raw Normal View History

package oscommands
2018-08-14 11:05:26 +02:00
import (
"bufio"
"fmt"
2018-12-05 19:33:46 +11:00
"io/ioutil"
2018-08-14 11:05:26 +02:00
"os"
"os/exec"
"path/filepath"
2018-10-17 20:38:13 +02:00
"regexp"
2018-08-16 17:04:39 -04:00
"strings"
"sync"
2018-08-14 11:05:26 +02:00
"github.com/go-errors/errors"
"github.com/atotto/clipboard"
"github.com/jesseduffield/lazygit/pkg/config"
"github.com/jesseduffield/lazygit/pkg/secureexec"
"github.com/jesseduffield/lazygit/pkg/utils"
2018-08-14 11:05:26 +02:00
"github.com/mgutz/str"
2018-08-23 14:22:03 +02:00
"github.com/sirupsen/logrus"
2018-08-14 11:05:26 +02:00
)
// Platform stores the os state
type Platform struct {
2021-03-01 23:16:48 +09:00
OS string
CatCmd string
Shell string
ShellArg string
EscapedQuote string
OpenCommand string
OpenLinkCommand string
2018-08-14 11:05:26 +02:00
}
// OSCommand holds all the os commands
type OSCommand struct {
Log *logrus.Entry
Platform *Platform
Config config.AppConfigurer
Command func(string, ...string) *exec.Cmd
BeforeExecuteCmd func(*exec.Cmd)
Getenv func(string) string
2018-08-14 11:05:26 +02:00
}
// NewOSCommand os command runner
func NewOSCommand(log *logrus.Entry, config config.AppConfigurer) *OSCommand {
2018-08-21 21:09:53 +02:00
return &OSCommand{
Log: log,
Platform: getPlatform(),
Config: config,
Command: secureexec.Command,
BeforeExecuteCmd: func(*exec.Cmd) {},
Getenv: os.Getenv,
2018-08-14 11:05:26 +02:00
}
}
// SetCommand sets the command function used by the struct.
// To be used for testing only
func (c *OSCommand) SetCommand(cmd func(string, ...string) *exec.Cmd) {
c.Command = cmd
}
func (c *OSCommand) SetBeforeExecuteCmd(cmd func(*exec.Cmd)) {
c.BeforeExecuteCmd = cmd
}
2020-03-21 18:27:20 +11:00
type RunCommandOptions struct {
EnvVars []string
}
func (c *OSCommand) RunCommandWithOutputWithOptions(command string, options RunCommandOptions) (string, error) {
c.Log.WithField("command", command).Info("RunCommand")
cmd := c.ExecutableFromString(command)
cmd.Env = append(cmd.Env, options.EnvVars...)
return sanitisedCommandOutput(cmd.CombinedOutput())
}
func (c *OSCommand) RunCommandWithOptions(command string, options RunCommandOptions) error {
_, err := c.RunCommandWithOutputWithOptions(command, options)
return err
}
2018-08-14 11:05:26 +02:00
// RunCommandWithOutput wrapper around commands returning their output and error
2019-11-21 22:17:18 +11:00
// NOTE: If you don't pass any formatArgs we'll just use the command directly,
// however there's a bizarre compiler error/warning when you pass in a formatString
// with a percent sign because it thinks it's supposed to be a formatString when
// in that case it's not. To get around that error you'll need to define the string
// in a variable and pass the variable into RunCommandWithOutput.
func (c *OSCommand) RunCommandWithOutput(formatString string, formatArgs ...interface{}) (string, error) {
command := formatString
2019-11-21 21:45:18 +11:00
if formatArgs != nil {
command = fmt.Sprintf(formatString, formatArgs...)
}
2018-08-14 11:05:26 +02:00
c.Log.WithField("command", command).Info("RunCommand")
cmd := c.ExecutableFromString(command)
output, err := sanitisedCommandOutput(cmd.CombinedOutput())
if err != nil {
c.Log.WithField("command", command).Error(err)
}
return output, err
}
// RunExecutableWithOutput runs an executable file and returns its output
func (c *OSCommand) RunExecutableWithOutput(cmd *exec.Cmd) (string, error) {
c.BeforeExecuteCmd(cmd)
return sanitisedCommandOutput(cmd.CombinedOutput())
}
// RunExecutable runs an executable file and returns an error if there was one
func (c *OSCommand) RunExecutable(cmd *exec.Cmd) error {
_, err := c.RunExecutableWithOutput(cmd)
return err
}
// ExecutableFromString takes a string like `git status` and returns an executable command for it
func (c *OSCommand) ExecutableFromString(commandStr string) *exec.Cmd {
splitCmd := str.ToArgv(commandStr)
cmd := c.Command(splitCmd[0], splitCmd[1:]...)
cmd.Env = append(os.Environ(), "GIT_OPTIONAL_LOCKS=0")
return cmd
2018-08-14 11:05:26 +02:00
}
// ShellCommandFromString takes a string like `git commit` and returns an executable shell command for it
func (c *OSCommand) ShellCommandFromString(commandStr string) *exec.Cmd {
quotedCommand := ""
// Windows does not seem to like quotes around the command
if c.Platform.OS == "windows" {
quotedCommand = commandStr
} else {
2021-03-01 23:16:48 +09:00
quotedCommand = c.Quote(commandStr)
}
shellCommand := fmt.Sprintf("%s %s %s", c.Platform.Shell, c.Platform.ShellArg, quotedCommand)
return c.ExecutableFromString(shellCommand)
}
// RunCommandWithOutputLive runs RunCommandWithOutputLiveWrapper
2018-12-12 22:10:33 +01:00
func (c *OSCommand) RunCommandWithOutputLive(command string, output func(string) string) error {
return RunCommandWithOutputLiveWrapper(c, command, output)
2018-10-17 20:38:13 +02:00
}
// DetectUnamePass detect a username / password / passphrase question in a command
// promptUserForCredential is a function that gets executed when this function detect you need to fillin a password or passphrase
// The promptUserForCredential argument will be "username", "password" or "passphrase" and expects the user's password/passphrase or username back
2020-08-11 21:43:55 +10:00
func (c *OSCommand) DetectUnamePass(command string, promptUserForCredential func(string) string) error {
2018-10-17 20:38:13 +02:00
ttyText := ""
2018-12-12 22:10:33 +01:00
errMessage := c.RunCommandWithOutputLive(command, func(word string) string {
2018-10-17 20:38:13 +02:00
ttyText = ttyText + " " + word
prompts := map[string]string{
`.+'s password:`: "password",
`Password\s*for\s*'.+':`: "password",
`Username\s*for\s*'.+':`: "username",
`Enter\s*passphrase\s*for\s*key\s*'.+':`: "passphrase",
2018-10-17 20:38:13 +02:00
}
2020-05-15 21:54:11 +10:00
for pattern, askFor := range prompts {
if match, _ := regexp.MatchString(pattern, ttyText); match {
2018-11-10 08:46:42 +01:00
ttyText = ""
2020-08-11 21:43:55 +10:00
return promptUserForCredential(askFor)
2018-11-10 08:46:42 +01:00
}
2018-10-17 20:38:13 +02:00
}
2018-11-10 08:46:42 +01:00
2018-10-17 20:38:13 +02:00
return ""
})
2018-12-12 22:10:33 +01:00
return errMessage
2018-10-17 20:38:13 +02:00
}
2018-08-14 11:05:26 +02:00
// RunCommand runs a command and just returns the error
func (c *OSCommand) RunCommand(formatString string, formatArgs ...interface{}) error {
_, err := c.RunCommandWithOutput(formatString, formatArgs...)
2018-08-14 11:05:26 +02:00
return err
}
2021-04-01 20:10:24 +11:00
// RunShellCommand runs shell commands i.e. 'sh -c <command>'. Good for when you
// need access to the shell
func (c *OSCommand) RunShellCommand(command string) error {
c.Log.WithField("command", command).Info("RunShellCommand")
cmd := c.Command(c.Platform.Shell, c.Platform.ShellArg, command)
_, err := sanitisedCommandOutput(cmd.CombinedOutput())
return err
}
2018-08-28 19:12:35 +10:00
// 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"
}
2018-08-14 11:05:26 +02:00
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
2018-09-01 13:27:58 +10:00
if outputString == "" {
return "", utils.WrapError(err)
2018-09-01 13:27:58 +10:00
}
2018-08-14 11:05:26 +02:00
return outputString, errors.New(outputString)
}
return outputString, nil
}
// OpenFile opens a file with the given
2018-08-21 22:17:34 +02:00
func (c *OSCommand) OpenFile(filename string) error {
2020-10-03 14:54:55 +10:00
commandTemplate := c.Config.GetUserConfig().OS.OpenCommand
templateValues := map[string]string{
"filename": c.Quote(filename),
2018-08-14 11:05:26 +02:00
}
2018-08-21 22:17:34 +02:00
command := utils.ResolvePlaceholderString(commandTemplate, templateValues)
2018-09-01 14:33:01 +10:00
err := c.RunCommand(command)
return err
2018-08-14 11:05:26 +02:00
}
2018-11-28 12:33:52 +13:00
// OpenLink opens a file with the given
func (c *OSCommand) OpenLink(link string) error {
2020-10-03 14:54:55 +10:00
commandTemplate := c.Config.GetUserConfig().OS.OpenLinkCommand
templateValues := map[string]string{
"link": c.Quote(link),
}
command := utils.ResolvePlaceholderString(commandTemplate, templateValues)
err := c.RunCommand(command)
return err
}
2018-08-14 11:05:26 +02:00
// PrepareSubProcess iniPrepareSubProcessrocess then tells the Gui to switch to it
// TODO: see if this needs to exist, given that ExecutableFromString does the same things
2018-08-21 22:33:25 +02:00
func (c *OSCommand) PrepareSubProcess(cmdName string, commandArgs ...string) *exec.Cmd {
cmd := c.Command(cmdName, commandArgs...)
if cmd != nil {
cmd.Env = append(os.Environ(), "GIT_OPTIONAL_LOCKS=0")
}
return cmd
2018-08-14 11:05:26 +02:00
}
2021-04-01 20:10:24 +11:00
// PrepareShellSubProcess returns the pointer to a custom command
func (c *OSCommand) PrepareShellSubProcess(command string) *exec.Cmd {
return c.PrepareSubProcess(c.Platform.Shell, c.Platform.ShellArg, command)
}
2018-08-14 11:05:26 +02:00
// Quote wraps a message in platform-specific quotation marks
func (c *OSCommand) Quote(message string) string {
2021-03-01 23:16:48 +09:00
if c.Platform.OS == "windows" {
message = strings.Replace(message, `"`, `"'"'"`, -1)
message = strings.Replace(message, `\"`, `\\"`, -1)
} else {
message = strings.Replace(message, `\`, `\\`, -1)
message = strings.Replace(message, `"`, `\"`, -1)
message = strings.Replace(message, "`", "\\`", -1)
message = strings.Replace(message, "$", "\\$", -1)
}
2021-03-01 23:16:48 +09:00
escapedQuote := c.Platform.EscapedQuote
return escapedQuote + message + escapedQuote
2018-08-14 11:05:26 +02:00
}
2018-08-18 20:14:44 +10:00
2018-08-21 08:40:51 +02:00
// AppendLineToFile adds a new line in file
func (c *OSCommand) AppendLineToFile(filename, line string) error {
2018-08-19 20:41:04 +10:00
f, err := os.OpenFile(filename, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
if err != nil {
return utils.WrapError(err)
2018-08-19 20:41:04 +10:00
}
defer f.Close()
_, err = f.WriteString("\n" + line)
2019-02-20 20:01:29 +11:00
if err != nil {
return utils.WrapError(err)
2019-02-20 20:01:29 +11:00
}
return nil
2018-08-19 20:41:04 +10:00
}
2018-12-05 19:33:46 +11:00
// CreateTempFile writes a string to a new temp file and returns the file's name
func (c *OSCommand) CreateTempFile(filename, content string) (string, error) {
tmpfile, err := ioutil.TempFile("", filename)
if err != nil {
c.Log.Error(err)
return "", utils.WrapError(err)
2018-12-05 19:33:46 +11:00
}
if _, err := tmpfile.WriteString(content); err != nil {
2018-12-05 19:33:46 +11:00
c.Log.Error(err)
return "", utils.WrapError(err)
2018-12-05 19:33:46 +11:00
}
if err := tmpfile.Close(); err != nil {
c.Log.Error(err)
return "", utils.WrapError(err)
2018-12-05 19:33:46 +11:00
}
return tmpfile.Name(), nil
}
// CreateFileWithContent creates a file with the given content
func (c *OSCommand) CreateFileWithContent(path string, content string) error {
if err := os.MkdirAll(filepath.Dir(path), os.ModePerm); err != nil {
c.Log.Error(err)
return err
}
if err := ioutil.WriteFile(path, []byte(content), 0644); err != nil {
c.Log.Error(err)
return utils.WrapError(err)
}
return nil
}
2019-03-18 20:44:33 +11:00
// Remove removes a file or directory at the specified path
func (c *OSCommand) Remove(filename string) error {
err := os.RemoveAll(filename)
return utils.WrapError(err)
2018-12-05 19:33:46 +11:00
}
// FileExists checks whether a file exists at the specified path
func (c *OSCommand) FileExists(path string) (bool, error) {
if _, err := os.Stat(path); err != nil {
if os.IsNotExist(err) {
return false, nil
}
return false, err
}
return true, nil
}
// RunPreparedCommand takes a pointer to an exec.Cmd and runs it
// this is useful if you need to give your command some environment variables
// before running it
func (c *OSCommand) RunPreparedCommand(cmd *exec.Cmd) error {
c.BeforeExecuteCmd(cmd)
out, err := cmd.CombinedOutput()
outString := string(out)
c.Log.Info(outString)
if err != nil {
if len(outString) == 0 {
return err
}
return errors.New(outString)
}
return nil
}
// GetLazygitPath returns the path of the currently executed file
func (c *OSCommand) GetLazygitPath() string {
ex, err := os.Executable() // get the executable path for git to use
if err != nil {
ex = os.Args[0] // fallback to the first call argument if needed
}
2020-01-07 18:25:00 +00:00
return `"` + filepath.ToSlash(ex) + `"`
}
2019-03-12 21:43:56 +11:00
// PipeCommands runs a heap of commands and pipes their inputs/outputs together like A | B | C
func (c *OSCommand) PipeCommands(commandStrings ...string) error {
cmds := make([]*exec.Cmd, len(commandStrings))
for i, str := range commandStrings {
cmds[i] = c.ExecutableFromString(str)
}
for i := 0; i < len(cmds)-1; i++ {
stdout, err := cmds[i].StdoutPipe()
if err != nil {
return err
}
cmds[i+1].Stdin = stdout
}
// keeping this here in case I adapt this code for some other purpose in the future
// cmds[len(cmds)-1].Stdout = os.Stdout
finalErrors := []string{}
wg := sync.WaitGroup{}
wg.Add(len(cmds))
for _, cmd := range cmds {
currentCmd := cmd
2020-10-07 21:19:38 +11:00
go utils.Safe(func() {
stderr, err := currentCmd.StderrPipe()
if err != nil {
c.Log.Error(err)
}
if err := currentCmd.Start(); err != nil {
c.Log.Error(err)
}
if b, err := ioutil.ReadAll(stderr); err == nil {
if len(b) > 0 {
finalErrors = append(finalErrors, string(b))
}
}
if err := currentCmd.Wait(); err != nil {
c.Log.Error(err)
}
wg.Done()
2020-10-07 21:19:38 +11:00
})
}
wg.Wait()
if len(finalErrors) > 0 {
return errors.New(strings.Join(finalErrors, "\n"))
}
return nil
}
2020-03-01 12:30:48 +11:00
func Kill(cmd *exec.Cmd) error {
if cmd.Process == nil {
// somebody got to it before we were able to, poor bastard
return nil
}
return cmd.Process.Kill()
}
func RunLineOutputCmd(cmd *exec.Cmd, onLine func(line string) (bool, error)) error {
stdoutPipe, err := cmd.StdoutPipe()
if err != nil {
return err
}
scanner := bufio.NewScanner(stdoutPipe)
scanner.Split(bufio.ScanLines)
if err := cmd.Start(); err != nil {
return err
}
for scanner.Scan() {
line := scanner.Text()
stop, err := onLine(line)
if err != nil {
return err
}
if stop {
2020-11-16 20:38:26 +11:00
_ = cmd.Process.Kill()
break
}
}
2020-11-16 20:38:26 +11:00
_ = cmd.Wait()
return nil
}
2020-04-15 20:30:24 +10:00
func (c *OSCommand) CopyToClipboard(str string) error {
return clipboard.WriteAll(str)
2020-04-15 20:30:24 +10:00
}