package git_commands import ( "errors" "fmt" "io/fs" "log" "os" ) type WorktreeCommands struct { *GitCommon } func NewWorktreeCommands(gitCommon *GitCommon) *WorktreeCommands { return &WorktreeCommands{ GitCommon: gitCommon, } } type NewWorktreeOpts struct { // required. The path of the new worktree. Path string // required. The base branch/ref. Base string // if true, ends up with a detached head Detach bool // optional. if empty, and if detach is false, we will checkout the base Branch string } func (self *WorktreeCommands) New(opts NewWorktreeOpts) error { if opts.Detach && opts.Branch != "" { panic("cannot specify branch when detaching") } cmdArgs := NewGitCmd("worktree").Arg("add"). ArgIf(opts.Detach, "--detach"). ArgIf(opts.Branch != "", "-b", opts.Branch). Arg(opts.Path, opts.Base) return self.cmd.New(cmdArgs.ToArgv()).Run() } func (self *WorktreeCommands) Delete(worktreePath string, force bool) error { cmdArgs := NewGitCmd("worktree").Arg("remove").ArgIf(force, "-f").Arg(worktreePath).ToArgv() return self.cmd.New(cmdArgs).Run() } func (self *WorktreeCommands) Detach(worktreePath string) error { cmdArgs := NewGitCmd("checkout").Arg("--detach").ToArgv() return self.cmd.New(cmdArgs).SetWd(worktreePath).Run() } func (self *WorktreeCommands) IsCurrentWorktree(path string) bool { return IsCurrentWorktree(path) } func IsCurrentWorktree(path string) bool { pwd, err := os.Getwd() if err != nil { log.Fatalln(err.Error()) } return EqualPath(pwd, path) } func (self *WorktreeCommands) IsWorktreePathMissing(path string) bool { if _, err := os.Stat(path); err != nil { if errors.Is(err, fs.ErrNotExist) { return true } log.Fatalln(fmt.Errorf("failed to check if worktree path `%s` exists\n%w", path, err).Error()) } return false } // checks if two paths are equal // TODO: support relative paths func EqualPath(a string, b string) bool { return a == b }