mirror of
https://github.com/jesseduffield/lazygit.git
synced 2025-01-26 05:37:18 +02:00
63dc07fded
By constructing an arg vector manually, we no longer need to quote arguments Mandate that args must be passed when building a command Now you need to provide an args array when building a command. There are a handful of places where we need to deal with a string, such as with user-defined custom commands, and for those we now require that at the callsite they use str.ToArgv to do that. I don't want to provide a method out of the box for it because I want to discourage its use. For some reason we were invoking a command through a shell when amending a commit, and I don't believe we needed to do that as there was nothing user- supplied about the command. So I've switched to using a regular command out- side the shell there
196 lines
4.7 KiB
Go
196 lines
4.7 KiB
Go
package git_commands
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/oscommands"
|
|
)
|
|
|
|
type StashCommands struct {
|
|
*GitCommon
|
|
fileLoader *FileLoader
|
|
workingTree *WorkingTreeCommands
|
|
}
|
|
|
|
func NewStashCommands(
|
|
gitCommon *GitCommon,
|
|
fileLoader *FileLoader,
|
|
workingTree *WorkingTreeCommands,
|
|
) *StashCommands {
|
|
return &StashCommands{
|
|
GitCommon: gitCommon,
|
|
fileLoader: fileLoader,
|
|
workingTree: workingTree,
|
|
}
|
|
}
|
|
|
|
func (self *StashCommands) DropNewest() error {
|
|
cmdArgs := NewGitCmd("stash").Arg("drop").ToArgv()
|
|
|
|
return self.cmd.New(cmdArgs).Run()
|
|
}
|
|
|
|
func (self *StashCommands) Drop(index int) error {
|
|
cmdArgs := NewGitCmd("stash").Arg("drop", fmt.Sprintf("stash@{%d}", index)).
|
|
ToArgv()
|
|
|
|
return self.cmd.New(cmdArgs).Run()
|
|
}
|
|
|
|
func (self *StashCommands) Pop(index int) error {
|
|
cmdArgs := NewGitCmd("stash").Arg("pop", fmt.Sprintf("stash@{%d}", index)).
|
|
ToArgv()
|
|
|
|
return self.cmd.New(cmdArgs).Run()
|
|
}
|
|
|
|
func (self *StashCommands) Apply(index int) error {
|
|
cmdArgs := NewGitCmd("stash").Arg("apply", fmt.Sprintf("stash@{%d}", index)).
|
|
ToArgv()
|
|
|
|
return self.cmd.New(cmdArgs).Run()
|
|
}
|
|
|
|
// Save save stash
|
|
func (self *StashCommands) Save(message string) error {
|
|
cmdArgs := NewGitCmd("stash").Arg("save", message).
|
|
ToArgv()
|
|
|
|
return self.cmd.New(cmdArgs).Run()
|
|
}
|
|
|
|
func (self *StashCommands) Store(sha string, message string) error {
|
|
trimmedMessage := strings.Trim(message, " \t")
|
|
|
|
cmdArgs := NewGitCmd("stash").Arg("store", sha).
|
|
ArgIf(trimmedMessage != "", "-m", trimmedMessage).
|
|
ToArgv()
|
|
|
|
return self.cmd.New(cmdArgs).Run()
|
|
}
|
|
|
|
func (self *StashCommands) Sha(index int) (string, error) {
|
|
cmdArgs := NewGitCmd("rev-parse").
|
|
Arg(fmt.Sprintf("refs/stash@{%d}", index)).
|
|
ToArgv()
|
|
|
|
sha, _, err := self.cmd.New(cmdArgs).DontLog().RunWithOutputs()
|
|
return strings.Trim(sha, "\r\n"), err
|
|
}
|
|
|
|
func (self *StashCommands) ShowStashEntryCmdObj(index int, ignoreWhitespace bool) oscommands.ICmdObj {
|
|
cmdArgs := NewGitCmd("stash").Arg("show").
|
|
Arg("-p").
|
|
Arg("--stat").
|
|
Arg(fmt.Sprintf("--color=%s", self.UserConfig.Git.Paging.ColorArg)).
|
|
Arg(fmt.Sprintf("--unified=%d", self.UserConfig.Git.DiffContextSize)).
|
|
ArgIf(ignoreWhitespace, "--ignore-all-space").
|
|
Arg(fmt.Sprintf("stash@{%d}", index)).
|
|
ToArgv()
|
|
|
|
return self.cmd.New(cmdArgs).DontLog()
|
|
}
|
|
|
|
func (self *StashCommands) StashAndKeepIndex(message string) error {
|
|
cmdArgs := NewGitCmd("stash").Arg("save", message, "--keep-index").
|
|
ToArgv()
|
|
|
|
return self.cmd.New(cmdArgs).Run()
|
|
}
|
|
|
|
func (self *StashCommands) StashUnstagedChanges(message string) error {
|
|
if err := self.cmd.New(
|
|
NewGitCmd("commit").
|
|
Arg("--no-verify", "-m", "[lazygit] stashing unstaged changes").
|
|
ToArgv(),
|
|
).Run(); err != nil {
|
|
return err
|
|
}
|
|
if err := self.Save(message); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := self.cmd.New(
|
|
NewGitCmd("reset").Arg("--soft", "HEAD^").ToArgv(),
|
|
).Run(); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// SaveStagedChanges stashes only the currently staged changes. This takes a few steps
|
|
// shoutouts to Joe on https://stackoverflow.com/questions/14759748/stashing-only-staged-changes-in-git-is-it-possible
|
|
func (self *StashCommands) SaveStagedChanges(message string) error {
|
|
// wrap in 'writing', which uses a mutex
|
|
if err := self.cmd.New(
|
|
NewGitCmd("stash").Arg("--keep-index").ToArgv(),
|
|
).Run(); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := self.Save(message); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := self.cmd.New(
|
|
NewGitCmd("stash").Arg("apply", "stash@{1}").ToArgv(),
|
|
).Run(); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := self.os.PipeCommands(
|
|
self.cmd.New(NewGitCmd("stash").Arg("show", "-p").ToArgv()),
|
|
self.cmd.New(NewGitCmd("apply").Arg("-R").ToArgv()),
|
|
); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := self.cmd.New(
|
|
NewGitCmd("stash").Arg("drop", "stash@{1}").ToArgv(),
|
|
).Run(); err != nil {
|
|
return err
|
|
}
|
|
|
|
// if you had staged an untracked file, that will now appear as 'AD' in git status
|
|
// meaning it's deleted in your working tree but added in your index. Given that it's
|
|
// now safely stashed, we need to remove it.
|
|
files := self.fileLoader.
|
|
GetStatusFiles(GetStatusFileOptions{})
|
|
|
|
for _, file := range files {
|
|
if file.ShortStatus == "AD" {
|
|
if err := self.workingTree.UnStageFile(file.Names(), false); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (self *StashCommands) StashIncludeUntrackedChanges(message string) error {
|
|
return self.cmd.New(
|
|
NewGitCmd("stash").Arg("save", message, "--include-untracked").
|
|
ToArgv(),
|
|
).Run()
|
|
}
|
|
|
|
func (self *StashCommands) Rename(index int, message string) error {
|
|
sha, err := self.Sha(index)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := self.Drop(index); err != nil {
|
|
return err
|
|
}
|
|
|
|
err = self.Store(sha, message)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|