2022-01-16 14:46:53 +11:00
|
|
|
package controllers
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
|
2023-07-09 11:32:27 +10:00
|
|
|
"github.com/jesseduffield/gocui"
|
2022-01-16 14:46:53 +11:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/git_commands"
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
2023-10-08 18:09:29 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/context"
|
2022-01-16 14:46:53 +11:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/types"
|
2023-07-26 17:36:14 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
2022-01-16 14:46:53 +11:00
|
|
|
)
|
|
|
|
|
|
|
|
type SyncController struct {
|
2022-02-05 14:42:56 +11:00
|
|
|
baseController
|
2023-03-23 18:47:29 +11:00
|
|
|
c *ControllerCommon
|
2022-01-16 14:46:53 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.IController = &SyncController{}
|
|
|
|
|
|
|
|
func NewSyncController(
|
2023-03-23 18:47:29 +11:00
|
|
|
common *ControllerCommon,
|
2022-01-16 14:46:53 +11:00
|
|
|
) *SyncController {
|
|
|
|
return &SyncController{
|
2023-03-23 18:47:29 +11:00
|
|
|
baseController: baseController{},
|
|
|
|
c: common,
|
2022-01-16 14:46:53 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-05 10:31:07 +11:00
|
|
|
func (self *SyncController) GetKeybindings(opts types.KeybindingsOpts) []*types.Binding {
|
2022-01-16 14:46:53 +11:00
|
|
|
bindings := []*types.Binding{
|
|
|
|
{
|
2023-02-19 11:42:00 +11:00
|
|
|
Key: opts.GetKey(opts.Config.Universal.Push),
|
2022-02-05 10:31:07 +11:00
|
|
|
Handler: opts.Guards.NoPopupPanel(self.HandlePush),
|
2023-05-25 21:11:51 +10:00
|
|
|
Description: self.c.Tr.Push,
|
2022-01-16 14:46:53 +11:00
|
|
|
},
|
|
|
|
{
|
2023-02-19 11:42:00 +11:00
|
|
|
Key: opts.GetKey(opts.Config.Universal.Pull),
|
2022-02-05 10:31:07 +11:00
|
|
|
Handler: opts.Guards.NoPopupPanel(self.HandlePull),
|
2023-05-25 21:11:51 +10:00
|
|
|
Description: self.c.Tr.Pull,
|
2022-01-16 14:46:53 +11:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
return bindings
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *SyncController) Context() types.Context {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *SyncController) HandlePush() error {
|
|
|
|
return self.branchCheckedOut(self.push)()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *SyncController) HandlePull() error {
|
|
|
|
return self.branchCheckedOut(self.pull)()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *SyncController) branchCheckedOut(f func(*models.Branch) error) func() error {
|
|
|
|
return func() error {
|
2023-03-23 18:47:29 +11:00
|
|
|
currentBranch := self.c.Helpers().Refs.GetCheckedOutRef()
|
2022-01-16 14:46:53 +11:00
|
|
|
if currentBranch == nil {
|
|
|
|
// need to wait for branches to refresh
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return f(currentBranch)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *SyncController) push(currentBranch *models.Branch) error {
|
|
|
|
// if we have pullables we'll ask if the user wants to force push
|
|
|
|
if currentBranch.IsTrackingRemote() {
|
2022-03-27 11:22:16 +11:00
|
|
|
opts := pushOpts{}
|
2022-01-16 14:46:53 +11:00
|
|
|
if currentBranch.HasCommitsToPull() {
|
2023-10-08 18:09:29 +02:00
|
|
|
return self.requestToForcePush(currentBranch, opts)
|
2022-01-16 14:46:53 +11:00
|
|
|
} else {
|
2023-10-08 18:09:29 +02:00
|
|
|
return self.pushAux(currentBranch, opts)
|
2022-01-16 14:46:53 +11:00
|
|
|
}
|
|
|
|
} else {
|
2023-03-23 13:04:57 +11:00
|
|
|
if self.c.Git().Config.GetPushToCurrent() {
|
2023-10-08 18:09:29 +02:00
|
|
|
return self.pushAux(currentBranch, pushOpts{setUpstream: true})
|
2022-01-16 14:46:53 +11:00
|
|
|
} else {
|
2023-03-23 18:47:29 +11:00
|
|
|
return self.c.Helpers().Upstream.PromptForUpstreamWithInitialContent(currentBranch, func(upstream string) error {
|
|
|
|
upstreamRemote, upstreamBranch, err := self.c.Helpers().Upstream.ParseUpstream(upstream)
|
2022-03-27 11:22:16 +11:00
|
|
|
if err != nil {
|
|
|
|
return self.c.Error(err)
|
2022-01-16 14:46:53 +11:00
|
|
|
}
|
|
|
|
|
2023-10-08 18:09:29 +02:00
|
|
|
return self.pushAux(currentBranch, pushOpts{
|
2022-03-27 11:22:16 +11:00
|
|
|
setUpstream: true,
|
2022-01-16 14:46:53 +11:00
|
|
|
upstreamRemote: upstreamRemote,
|
|
|
|
upstreamBranch: upstreamBranch,
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *SyncController) pull(currentBranch *models.Branch) error {
|
|
|
|
action := self.c.Tr.Actions.Pull
|
|
|
|
|
|
|
|
// if we have no upstream branch we need to set that first
|
|
|
|
if !currentBranch.IsTrackingRemote() {
|
2023-03-23 18:47:29 +11:00
|
|
|
return self.c.Helpers().Upstream.PromptForUpstreamWithInitialContent(currentBranch, func(upstream string) error {
|
2022-03-27 11:22:16 +11:00
|
|
|
if err := self.setCurrentBranchUpstream(upstream); err != nil {
|
|
|
|
return self.c.Error(err)
|
2022-01-16 14:46:53 +11:00
|
|
|
}
|
|
|
|
|
2023-10-08 18:09:29 +02:00
|
|
|
return self.PullAux(currentBranch, PullFilesOptions{Action: action})
|
2022-01-16 14:46:53 +11:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-10-08 18:09:29 +02:00
|
|
|
return self.PullAux(currentBranch, PullFilesOptions{Action: action})
|
2022-03-27 11:22:16 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
func (self *SyncController) setCurrentBranchUpstream(upstream string) error {
|
2023-03-23 18:47:29 +11:00
|
|
|
upstreamRemote, upstreamBranch, err := self.c.Helpers().Upstream.ParseUpstream(upstream)
|
2022-03-27 11:22:16 +11:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-03-23 13:04:57 +11:00
|
|
|
if err := self.c.Git().Branch.SetCurrentBranchUpstream(upstreamRemote, upstreamBranch); err != nil {
|
2022-03-27 11:22:16 +11:00
|
|
|
if strings.Contains(err.Error(), "does not exist") {
|
|
|
|
return fmt.Errorf(
|
|
|
|
"upstream branch %s/%s not found.\nIf you expect it to exist, you should fetch (with 'f').\nOtherwise, you should push (with 'shift+P')",
|
|
|
|
upstreamRemote, upstreamBranch,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-01-16 14:46:53 +11:00
|
|
|
type PullFilesOptions struct {
|
|
|
|
UpstreamRemote string
|
|
|
|
UpstreamBranch string
|
|
|
|
FastForwardOnly bool
|
|
|
|
Action string
|
|
|
|
}
|
|
|
|
|
2023-10-08 18:09:29 +02:00
|
|
|
func (self *SyncController) PullAux(currentBranch *models.Branch, opts PullFilesOptions) error {
|
|
|
|
return self.c.WithInlineStatus(currentBranch, types.ItemOperationPulling, context.LOCAL_BRANCHES_CONTEXT_KEY, func(task gocui.Task) error {
|
2023-07-09 11:32:27 +10:00
|
|
|
return self.pullWithLock(task, opts)
|
2022-01-16 14:46:53 +11:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-07-09 21:09:52 +10:00
|
|
|
func (self *SyncController) pullWithLock(task gocui.Task, opts PullFilesOptions) error {
|
2022-01-16 14:46:53 +11:00
|
|
|
self.c.LogAction(opts.Action)
|
|
|
|
|
2023-03-23 13:04:57 +11:00
|
|
|
err := self.c.Git().Sync.Pull(
|
2023-07-09 11:32:27 +10:00
|
|
|
task,
|
2022-01-16 14:46:53 +11:00
|
|
|
git_commands.PullOptions{
|
|
|
|
RemoteName: opts.UpstreamRemote,
|
|
|
|
BranchName: opts.UpstreamBranch,
|
|
|
|
FastForwardOnly: opts.FastForwardOnly,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
2023-03-23 18:47:29 +11:00
|
|
|
return self.c.Helpers().MergeAndRebase.CheckMergeOrRebase(err)
|
2022-01-16 14:46:53 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
type pushOpts struct {
|
|
|
|
force bool
|
|
|
|
upstreamRemote string
|
|
|
|
upstreamBranch string
|
|
|
|
setUpstream bool
|
|
|
|
}
|
|
|
|
|
2023-10-08 18:09:29 +02:00
|
|
|
func (self *SyncController) pushAux(currentBranch *models.Branch, opts pushOpts) error {
|
|
|
|
return self.c.WithInlineStatus(currentBranch, types.ItemOperationPushing, context.LOCAL_BRANCHES_CONTEXT_KEY, func(task gocui.Task) error {
|
2022-01-16 14:46:53 +11:00
|
|
|
self.c.LogAction(self.c.Tr.Actions.Push)
|
2023-07-09 11:32:27 +10:00
|
|
|
err := self.c.Git().Sync.Push(
|
|
|
|
task,
|
|
|
|
git_commands.PushOpts{
|
|
|
|
Force: opts.force,
|
|
|
|
UpstreamRemote: opts.upstreamRemote,
|
|
|
|
UpstreamBranch: opts.upstreamBranch,
|
|
|
|
SetUpstream: opts.setUpstream,
|
|
|
|
})
|
2022-01-16 14:46:53 +11:00
|
|
|
if err != nil {
|
|
|
|
if !opts.force && strings.Contains(err.Error(), "Updates were rejected") {
|
|
|
|
forcePushDisabled := self.c.UserConfig.Git.DisableForcePushing
|
|
|
|
if forcePushDisabled {
|
|
|
|
_ = self.c.ErrorMsg(self.c.Tr.UpdatesRejectedAndForcePushDisabled)
|
|
|
|
return nil
|
|
|
|
}
|
2022-03-30 08:48:29 +02:00
|
|
|
_ = self.c.Confirm(types.ConfirmOpts{
|
2022-01-16 14:46:53 +11:00
|
|
|
Title: self.c.Tr.ForcePush,
|
2023-07-26 17:36:14 +02:00
|
|
|
Prompt: self.forcePushPrompt(),
|
2022-01-16 14:46:53 +11:00
|
|
|
HandleConfirm: func() error {
|
|
|
|
newOpts := opts
|
|
|
|
newOpts.force = true
|
|
|
|
|
2023-10-08 18:09:29 +02:00
|
|
|
return self.pushAux(currentBranch, newOpts)
|
2022-01-16 14:46:53 +11:00
|
|
|
},
|
|
|
|
})
|
|
|
|
return nil
|
|
|
|
}
|
2023-10-08 18:09:29 +02:00
|
|
|
return err
|
2022-01-16 14:46:53 +11:00
|
|
|
}
|
|
|
|
return self.c.Refresh(types.RefreshOptions{Mode: types.ASYNC})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-10-08 18:09:29 +02:00
|
|
|
func (self *SyncController) requestToForcePush(currentBranch *models.Branch, opts pushOpts) error {
|
2022-01-16 14:46:53 +11:00
|
|
|
forcePushDisabled := self.c.UserConfig.Git.DisableForcePushing
|
|
|
|
if forcePushDisabled {
|
|
|
|
return self.c.ErrorMsg(self.c.Tr.ForcePushDisabled)
|
|
|
|
}
|
|
|
|
|
2022-03-30 08:48:29 +02:00
|
|
|
return self.c.Confirm(types.ConfirmOpts{
|
2022-01-16 14:46:53 +11:00
|
|
|
Title: self.c.Tr.ForcePush,
|
2023-07-26 17:36:14 +02:00
|
|
|
Prompt: self.forcePushPrompt(),
|
2022-01-16 14:46:53 +11:00
|
|
|
HandleConfirm: func() error {
|
2022-03-27 11:22:16 +11:00
|
|
|
opts.force = true
|
2023-10-08 18:09:29 +02:00
|
|
|
return self.pushAux(currentBranch, opts)
|
2022-01-16 14:46:53 +11:00
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
2023-07-26 17:36:14 +02:00
|
|
|
|
|
|
|
func (self *SyncController) forcePushPrompt() string {
|
|
|
|
return utils.ResolvePlaceholderString(
|
|
|
|
self.c.Tr.ForcePushPrompt,
|
|
|
|
map[string]string{
|
|
|
|
"cancelKey": self.c.UserConfig.Keybinding.Universal.Return,
|
|
|
|
"confirmKey": self.c.UserConfig.Keybinding.Universal.Confirm,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|