2022-01-08 05:00:36 +02:00
|
|
|
package git_commands
|
2020-09-29 12:03:39 +02:00
|
|
|
|
2021-12-30 08:19:01 +02:00
|
|
|
import (
|
2022-10-15 04:15:31 +02:00
|
|
|
"errors"
|
2021-12-30 08:19:01 +02:00
|
|
|
"fmt"
|
2022-10-15 04:15:31 +02:00
|
|
|
"regexp"
|
2022-10-14 15:19:53 +02:00
|
|
|
"strings"
|
2021-12-30 08:19:01 +02:00
|
|
|
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/loaders"
|
2022-01-05 02:57:32 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/oscommands"
|
2021-12-30 08:19:01 +02:00
|
|
|
)
|
2020-09-29 12:03:39 +02:00
|
|
|
|
2022-01-02 01:34:33 +02:00
|
|
|
type StashCommands struct {
|
2022-01-18 12:26:21 +02:00
|
|
|
*GitCommon
|
2022-01-02 01:34:33 +02:00
|
|
|
fileLoader *loaders.FileLoader
|
|
|
|
workingTree *WorkingTreeCommands
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewStashCommands(
|
2022-01-18 12:26:21 +02:00
|
|
|
gitCommon *GitCommon,
|
2022-01-02 01:34:33 +02:00
|
|
|
fileLoader *loaders.FileLoader,
|
|
|
|
workingTree *WorkingTreeCommands,
|
|
|
|
) *StashCommands {
|
|
|
|
return &StashCommands{
|
2022-01-18 12:26:21 +02:00
|
|
|
GitCommon: gitCommon,
|
2022-01-02 01:34:33 +02:00
|
|
|
fileLoader: fileLoader,
|
|
|
|
workingTree: workingTree,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-08 13:30:43 +02:00
|
|
|
func (self *StashCommands) DropNewest() error {
|
|
|
|
return self.cmd.New("git stash drop").Run()
|
|
|
|
}
|
|
|
|
|
2022-10-14 15:19:53 +02:00
|
|
|
func (self *StashCommands) Drop(index int) (string, error) {
|
|
|
|
output, _, err := self.cmd.New(fmt.Sprintf("git stash drop stash@{%d}", index)).RunWithOutputs()
|
|
|
|
return output, err
|
2022-01-02 01:34:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (self *StashCommands) Pop(index int) error {
|
|
|
|
return self.cmd.New(fmt.Sprintf("git stash pop stash@{%d}", index)).Run()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *StashCommands) Apply(index int) error {
|
|
|
|
return self.cmd.New(fmt.Sprintf("git stash apply stash@{%d}", index)).Run()
|
2020-09-29 12:03:39 +02:00
|
|
|
}
|
|
|
|
|
2022-01-02 01:34:33 +02:00
|
|
|
// Save save stash
|
|
|
|
func (self *StashCommands) Save(message string) error {
|
|
|
|
return self.cmd.New("git stash save " + self.cmd.Quote(message)).Run()
|
2022-10-14 15:19:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (self *StashCommands) Store(sha string, message string) error {
|
|
|
|
trimmedMessage := strings.Trim(message, " \t")
|
|
|
|
if len(trimmedMessage) > 0 {
|
2022-10-15 04:03:47 +02:00
|
|
|
return self.cmd.New(fmt.Sprintf("git stash store %s -m %s", self.cmd.Quote(sha), self.cmd.Quote(trimmedMessage))).Run()
|
2022-10-14 15:19:53 +02:00
|
|
|
}
|
|
|
|
return self.cmd.New(fmt.Sprintf("git stash store %s", self.cmd.Quote(sha))).Run()
|
2020-09-29 12:03:39 +02:00
|
|
|
}
|
|
|
|
|
2022-01-02 01:34:33 +02:00
|
|
|
func (self *StashCommands) ShowStashEntryCmdObj(index int) oscommands.ICmdObj {
|
|
|
|
cmdStr := fmt.Sprintf("git stash show -p --stat --color=%s --unified=%d stash@{%d}", self.UserConfig.Git.Paging.ColorArg, self.UserConfig.Git.DiffContextSize, index)
|
2022-01-05 02:57:32 +02:00
|
|
|
|
2022-01-02 01:34:33 +02:00
|
|
|
return self.cmd.New(cmdStr).DontLog()
|
2020-09-29 12:03:39 +02:00
|
|
|
}
|
|
|
|
|
2022-04-08 11:32:23 +02:00
|
|
|
func (self *StashCommands) StashAndKeepIndex(message string) error {
|
|
|
|
return self.cmd.New(fmt.Sprintf("git stash save %s --keep-index", self.cmd.Quote(message))).Run()
|
|
|
|
}
|
|
|
|
|
2022-04-14 21:45:55 +02:00
|
|
|
func (self *StashCommands) StashUnstagedChanges(message string) error {
|
2022-04-15 08:03:25 +02:00
|
|
|
if err := self.cmd.New("git commit --no-verify -m \"[lazygit] stashing unstaged changes\"").Run(); err != nil {
|
2022-04-14 21:45:55 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := self.Save(message); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := self.cmd.New("git reset --soft HEAD^").Run(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-01-02 01:34:33 +02:00
|
|
|
// SaveStagedChanges stashes only the currently staged changes. This takes a few steps
|
2020-09-29 12:03:39 +02:00
|
|
|
// shoutouts to Joe on https://stackoverflow.com/questions/14759748/stashing-only-staged-changes-in-git-is-it-possible
|
2022-01-02 01:34:33 +02:00
|
|
|
func (self *StashCommands) SaveStagedChanges(message string) error {
|
2021-04-10 08:01:46 +02:00
|
|
|
// wrap in 'writing', which uses a mutex
|
2022-01-02 01:34:33 +02:00
|
|
|
if err := self.cmd.New("git stash --keep-index").Run(); err != nil {
|
2020-09-29 12:03:39 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-01-02 01:34:33 +02:00
|
|
|
if err := self.Save(message); err != nil {
|
2020-09-29 12:03:39 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-01-02 01:34:33 +02:00
|
|
|
if err := self.cmd.New("git stash apply stash@{1}").Run(); err != nil {
|
2020-09-29 12:03:39 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-01-18 12:26:21 +02:00
|
|
|
if err := self.os.PipeCommands("git stash show -p", "git apply -R"); err != nil {
|
2020-09-29 12:03:39 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-01-02 01:34:33 +02:00
|
|
|
if err := self.cmd.New("git stash drop stash@{1}").Run(); err != nil {
|
2020-09-29 12:03:39 +02:00
|
|
|
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.
|
2022-01-02 01:34:33 +02:00
|
|
|
files := self.fileLoader.
|
2021-12-30 08:19:01 +02:00
|
|
|
GetStatusFiles(loaders.GetStatusFileOptions{})
|
|
|
|
|
2020-09-29 12:03:39 +02:00
|
|
|
for _, file := range files {
|
|
|
|
if file.ShortStatus == "AD" {
|
2022-01-02 01:34:33 +02:00
|
|
|
if err := self.workingTree.UnStageFile(file.Names(), false); err != nil {
|
2020-09-29 12:03:39 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2022-10-15 04:15:31 +02:00
|
|
|
|
|
|
|
func (self *StashCommands) Rename(index int, message string) error {
|
|
|
|
output, err := self.Drop(index)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// `output` is in the following format:
|
|
|
|
// Dropped refs/stash@{0} (f0d0f20f2f61ffd6d6bfe0752deffa38845a3edd)
|
|
|
|
stashShaPattern := regexp.MustCompile(`\(([0-9a-f]+)\)`)
|
|
|
|
matches := stashShaPattern.FindStringSubmatch(output)
|
|
|
|
if len(matches) <= 1 {
|
|
|
|
return errors.New("Output of `git stash drop` is invalid") // Usually this error does not occur
|
|
|
|
}
|
|
|
|
stashSha := matches[1]
|
|
|
|
|
|
|
|
err = self.Store(stashSha, message)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|