mirror of
https://github.com/jesseduffield/lazygit.git
synced 2025-01-20 05:19:24 +02:00
178 lines
4.9 KiB
Go
178 lines
4.9 KiB
Go
package commands
|
|
|
|
import (
|
|
"errors"
|
|
"os"
|
|
"os/exec"
|
|
"runtime"
|
|
"regexp"
|
|
|
|
"github.com/davecgh/go-spew/spew"
|
|
|
|
"github.com/mgutz/str"
|
|
|
|
"github.com/Sirupsen/logrus"
|
|
gitconfig "github.com/tcnksm/go-gitconfig"
|
|
)
|
|
|
|
var (
|
|
// ErrNoOpenCommand : When we don't know which command to use to open a file
|
|
ErrNoOpenCommand = errors.New("Unsure what command to use to open this file")
|
|
// ErrNoEditorDefined : When we can't find an editor to edit a file
|
|
ErrNoEditorDefined = errors.New("No editor defined in $VISUAL, $EDITOR, or git config")
|
|
)
|
|
|
|
// Platform stores the os state
|
|
type Platform struct {
|
|
os string
|
|
shell string
|
|
shellArg string
|
|
escapedQuote string
|
|
}
|
|
|
|
// OSCommand holds all the os commands
|
|
type OSCommand struct {
|
|
Log *logrus.Logger
|
|
Platform *Platform
|
|
}
|
|
|
|
// NewOSCommand os command runner
|
|
func NewOSCommand(log *logrus.Logger) (*OSCommand, error) {
|
|
osCommand := &OSCommand{
|
|
Log: log,
|
|
Platform: getPlatform(),
|
|
}
|
|
return osCommand, nil
|
|
}
|
|
|
|
// 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)
|
|
cmdOut, err := exec.Command(splitCmd[0], splitCmd[1:]...).CombinedOutput()
|
|
return sanitisedCommandOutput(cmdOut, err)
|
|
}
|
|
|
|
// RunCommand runs a command and just returns the error
|
|
func (c *OSCommand) RunCommand(command string) error {
|
|
_, err := c.RunCommandWithOutput(command)
|
|
return err
|
|
}
|
|
|
|
// RunDirectCommand wrapper around direct commands
|
|
func (c *OSCommand) RunDirectCommand(command string) (string, error) {
|
|
c.Log.WithField("command", command).Info("RunDirectCommand")
|
|
args := str.ToArgv(c.Platform.shellArg + " " + command)
|
|
c.Log.Info(spew.Sdump(args))
|
|
|
|
cmdOut, err := exec.
|
|
Command(c.Platform.shell, args...).
|
|
CombinedOutput()
|
|
return sanitisedCommandOutput(cmdOut, err)
|
|
}
|
|
|
|
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
|
|
return outputString, errors.New(outputString)
|
|
}
|
|
return outputString, nil
|
|
}
|
|
|
|
func getPlatform() *Platform {
|
|
switch runtime.GOOS {
|
|
case "windows":
|
|
return &Platform{
|
|
os: "windows",
|
|
shell: "cmd",
|
|
shellArg: "/c",
|
|
escapedQuote: "\\\"",
|
|
}
|
|
default:
|
|
return &Platform{
|
|
os: runtime.GOOS,
|
|
shell: "bash",
|
|
shellArg: "-c",
|
|
escapedQuote: "\"",
|
|
}
|
|
}
|
|
}
|
|
|
|
// GetOpenCommand get open command
|
|
func (c *OSCommand) GetOpenCommand() (string, string, error) {
|
|
//NextStep open equivalents: xdg-open (linux), cygstart (cygwin), open (OSX)
|
|
trailMap := map[string]string{
|
|
"xdg-open": " &>/dev/null &",
|
|
"cygstart": "",
|
|
"open": "",
|
|
}
|
|
for name, trail := range trailMap {
|
|
if err := c.RunCommand("which " + name); err == nil {
|
|
return name, trail, nil
|
|
}
|
|
}
|
|
return "", "", ErrNoOpenCommand
|
|
}
|
|
|
|
// VsCodeOpenFile opens the file in code, with the -r flag to open in the
|
|
// current window
|
|
// each of these open files needs to have the same function signature because
|
|
// they're being passed as arguments into another function,
|
|
// but only editFile actually returns a *exec.Cmd
|
|
func (c *OSCommand) VsCodeOpenFile(filename string) (*exec.Cmd, error) {
|
|
return nil, c.RunCommand("code -r " + filename)
|
|
}
|
|
|
|
// SublimeOpenFile opens the filein sublime
|
|
// may be deprecated in the future
|
|
func (c *OSCommand) SublimeOpenFile(filename string) (*exec.Cmd, error) {
|
|
return nil, c.RunCommand("subl " + filename)
|
|
}
|
|
|
|
// OpenFile opens a file with the given
|
|
func (c *OSCommand) OpenFile(filename string) (*exec.Cmd, error) {
|
|
cmdName, cmdTrail, err := c.GetOpenCommand()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
err = c.RunCommand(cmdName + " " + filename + cmdTrail) // TODO: test on linux
|
|
return nil, 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, _ := gitconfig.Global("core.editor")
|
|
if editor == "" {
|
|
editor = os.Getenv("VISUAL")
|
|
}
|
|
if editor == "" {
|
|
editor = os.Getenv("EDITOR")
|
|
}
|
|
if editor == "" {
|
|
if err := c.RunCommand("which vi"); err == nil {
|
|
editor = "vi"
|
|
}
|
|
}
|
|
if editor == "" {
|
|
return nil, ErrNoEditorDefined
|
|
}
|
|
return c.PrepareSubProcess(editor, filename)
|
|
}
|
|
|
|
// PrepareSubProcess iniPrepareSubProcessrocess then tells the Gui to switch to it
|
|
func (c *OSCommand) PrepareSubProcess(cmdName string, commandArgs ...string) (*exec.Cmd, error) {
|
|
subprocess := exec.Command(cmdName, commandArgs...)
|
|
return subprocess, nil
|
|
}
|
|
|
|
// Quote wraps a message in platform-specific quotation marks
|
|
func (c *OSCommand) Quote(message string) string {
|
|
r := regexp.MustCompile("`")
|
|
message = r.ReplaceAllString(message, "\\`")
|
|
return c.Platform.escapedQuote + message + c.Platform.escapedQuote
|
|
}
|