2023-04-02 19:16:28 +02:00
|
|
|
package utils
|
|
|
|
|
|
|
|
import (
|
2023-04-04 10:23:50 +02:00
|
|
|
"fmt"
|
2023-04-02 19:16:28 +02:00
|
|
|
"os"
|
2023-04-04 10:23:50 +02:00
|
|
|
"strings"
|
2023-04-02 19:16:28 +02:00
|
|
|
|
|
|
|
"github.com/fsmiamoto/git-todo-parser/todo"
|
2023-04-04 10:23:50 +02:00
|
|
|
"github.com/samber/lo"
|
2023-04-02 19:16:28 +02:00
|
|
|
)
|
|
|
|
|
2023-04-06 07:04:25 +02:00
|
|
|
// Read a git-rebase-todo file, change the action for the given sha to
|
|
|
|
// newAction, and write it back
|
|
|
|
func EditRebaseTodo(filePath string, sha string, oldAction todo.TodoCommand, newAction todo.TodoCommand) error {
|
|
|
|
todos, err := ReadRebaseTodoFile(filePath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := range todos {
|
|
|
|
t := &todos[i]
|
|
|
|
// Comparing just the sha is not enough; we need to compare both the
|
|
|
|
// action and the sha, as the sha could appear multiple times (e.g. in a
|
|
|
|
// pick and later in a merge)
|
|
|
|
if t.Command == oldAction && equalShas(t.Commit, sha) {
|
|
|
|
t.Command = newAction
|
|
|
|
return WriteRebaseTodoFile(filePath, todos)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Should never get here
|
|
|
|
return fmt.Errorf("Todo %s not found in git-rebase-todo", sha)
|
|
|
|
}
|
|
|
|
|
2023-04-04 10:23:50 +02:00
|
|
|
func equalShas(a, b string) bool {
|
|
|
|
return strings.HasPrefix(a, b) || strings.HasPrefix(b, a)
|
|
|
|
}
|
|
|
|
|
2023-04-02 19:16:28 +02:00
|
|
|
func ReadRebaseTodoFile(fileName string) ([]todo.Todo, error) {
|
|
|
|
f, err := os.Open(fileName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
todos, err := todo.Parse(f)
|
|
|
|
err2 := f.Close()
|
|
|
|
if err == nil {
|
|
|
|
err = err2
|
|
|
|
}
|
|
|
|
return todos, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func WriteRebaseTodoFile(fileName string, todos []todo.Todo) error {
|
|
|
|
f, err := os.Create(fileName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = todo.Write(f, todos)
|
|
|
|
err2 := f.Close()
|
|
|
|
if err == nil {
|
|
|
|
err = err2
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
2023-04-04 10:23:50 +02:00
|
|
|
|
2023-04-06 11:13:42 +02:00
|
|
|
func PrependStrToTodoFile(filePath string, linesToPrepend []byte) error {
|
|
|
|
existingContent, err := os.ReadFile(filePath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
linesToPrepend = append(linesToPrepend, existingContent...)
|
|
|
|
return os.WriteFile(filePath, linesToPrepend, 0o644)
|
|
|
|
}
|
|
|
|
|
2023-04-04 10:23:50 +02:00
|
|
|
func MoveTodoDown(fileName string, sha string, action todo.TodoCommand) error {
|
|
|
|
todos, err := ReadRebaseTodoFile(fileName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
rearrangedTodos, err := moveTodoDown(todos, sha, action)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return WriteRebaseTodoFile(fileName, rearrangedTodos)
|
|
|
|
}
|
|
|
|
|
|
|
|
func MoveTodoUp(fileName string, sha string, action todo.TodoCommand) error {
|
|
|
|
todos, err := ReadRebaseTodoFile(fileName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
rearrangedTodos, err := moveTodoUp(todos, sha, action)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return WriteRebaseTodoFile(fileName, rearrangedTodos)
|
|
|
|
}
|
|
|
|
|
|
|
|
func moveTodoDown(todos []todo.Todo, sha string, action todo.TodoCommand) ([]todo.Todo, error) {
|
|
|
|
rearrangedTodos, err := moveTodoUp(lo.Reverse(todos), sha, action)
|
|
|
|
return lo.Reverse(rearrangedTodos), err
|
|
|
|
}
|
|
|
|
|
|
|
|
func moveTodoUp(todos []todo.Todo, sha string, action todo.TodoCommand) ([]todo.Todo, error) {
|
|
|
|
_, sourceIdx, ok := lo.FindIndexOf(todos, func(t todo.Todo) bool {
|
|
|
|
// Comparing just the sha is not enough; we need to compare both the
|
|
|
|
// action and the sha, as the sha could appear multiple times (e.g. in a
|
|
|
|
// pick and later in a merge)
|
|
|
|
return t.Command == action && equalShas(t.Commit, sha)
|
|
|
|
})
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
// Should never happen
|
|
|
|
return []todo.Todo{}, fmt.Errorf("Todo %s not found in git-rebase-todo", sha)
|
|
|
|
}
|
|
|
|
|
|
|
|
// The todos are ordered backwards compared to our model commits, so
|
|
|
|
// actually move the commit _down_ in the todos slice (i.e. towards
|
|
|
|
// the end of the slice)
|
|
|
|
|
|
|
|
// Find the next todo that we show in lazygit's commits view (skipping the rest)
|
|
|
|
_, skip, ok := lo.FindIndexOf(todos[sourceIdx+1:], isRenderedTodo)
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
// We expect callers to guard against this
|
|
|
|
return []todo.Todo{}, fmt.Errorf("Destination position for moving todo is out of range")
|
|
|
|
}
|
|
|
|
|
|
|
|
destinationIdx := sourceIdx + 1 + skip
|
|
|
|
|
|
|
|
rearrangedTodos := MoveElement(todos, sourceIdx, destinationIdx)
|
|
|
|
|
|
|
|
return rearrangedTodos, nil
|
|
|
|
}
|
|
|
|
|
2023-04-19 08:19:17 +02:00
|
|
|
func MoveFixupCommitDown(fileName string, originalSha string, fixupSha string) error {
|
|
|
|
todos, err := ReadRebaseTodoFile(fileName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-04-24 03:42:23 +02:00
|
|
|
newTodos, err := moveFixupCommitDown(todos, originalSha, fixupSha)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-04-19 08:19:17 +02:00
|
|
|
|
2023-04-24 03:42:23 +02:00
|
|
|
return WriteRebaseTodoFile(fileName, newTodos)
|
|
|
|
}
|
|
|
|
|
|
|
|
func moveFixupCommitDown(todos []todo.Todo, originalSha string, fixupSha string) ([]todo.Todo, error) {
|
|
|
|
isOriginal := func(t todo.Todo) bool {
|
|
|
|
return t.Command == todo.Pick && equalShas(t.Commit, originalSha)
|
2023-04-19 08:19:17 +02:00
|
|
|
}
|
|
|
|
|
2023-04-24 03:42:23 +02:00
|
|
|
isFixup := func(t todo.Todo) bool {
|
|
|
|
return t.Command == todo.Pick && equalShas(t.Commit, fixupSha)
|
2023-04-19 08:19:17 +02:00
|
|
|
}
|
|
|
|
|
2023-04-24 03:42:23 +02:00
|
|
|
originalShaCount := lo.CountBy(todos, isOriginal)
|
|
|
|
if originalShaCount != 1 {
|
|
|
|
return nil, fmt.Errorf("Expected exactly one original SHA, found %d", originalShaCount)
|
|
|
|
}
|
|
|
|
|
|
|
|
fixupShaCount := lo.CountBy(todos, isFixup)
|
|
|
|
if fixupShaCount != 1 {
|
|
|
|
return nil, fmt.Errorf("Expected exactly one fixup SHA, found %d", fixupShaCount)
|
|
|
|
}
|
|
|
|
|
|
|
|
_, fixupIndex, _ := lo.FindIndexOf(todos, isFixup)
|
|
|
|
_, originalIndex, _ := lo.FindIndexOf(todos, isOriginal)
|
|
|
|
|
|
|
|
newTodos := MoveElement(todos, fixupIndex, originalIndex+1)
|
|
|
|
|
|
|
|
newTodos[originalIndex+1].Command = todo.Fixup
|
|
|
|
|
|
|
|
return newTodos, nil
|
2023-04-19 08:19:17 +02:00
|
|
|
}
|
|
|
|
|
2023-04-04 10:23:50 +02:00
|
|
|
// We render a todo in the commits view if it's a commit or if it's an
|
|
|
|
// update-ref. We don't render label, reset, or comment lines.
|
|
|
|
func isRenderedTodo(t todo.Todo) bool {
|
|
|
|
return t.Commit != "" || t.Command == todo.UpdateRef
|
|
|
|
}
|