Skip to content

Commit

Permalink
Fix bug when pushing into different branches
Browse files Browse the repository at this point in the history
Changes replacing go-git with git2go introduced a bug
in which pushes into new branches squashes all commits
into one.

xref: #324
Signed-off-by: Paulo Gomes <[email protected]>
  • Loading branch information
Paulo Gomes committed Mar 23, 2022
1 parent 65fc610 commit 4b35fe3
Show file tree
Hide file tree
Showing 3 changed files with 93 additions and 48 deletions.
3 changes: 3 additions & 0 deletions controllers/git_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -134,6 +134,9 @@ func TestPushRejected(t *testing.T) {

repoURL := gitServer.HTTPAddressWithCredentials() + "/appconfig.git"
repo, err := clone(repoURL, "origin", "main")
if err != nil {
t.Fatal(err)
}

// This is here to guard against push in general being broken
err = push(context.TODO(), repo.Workdir(), "main", repoAccess{
Expand Down
104 changes: 56 additions & 48 deletions controllers/imageupdateautomation_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -160,7 +160,7 @@ func (r *ImageUpdateAutomationReconciler) Reconcile(ctx context.Context, req ctr

// only GitRepository objects are supported for now
if kind := auto.Spec.SourceRef.Kind; kind != sourcev1.GitRepositoryKind {
return failWithError(fmt.Errorf("source kind %q not supported", kind))
return failWithError(fmt.Errorf("source kind '%s' not supported", kind))
}

gitSpec := auto.Spec.GitSpec
Expand Down Expand Up @@ -292,10 +292,7 @@ func (r *ImageUpdateAutomationReconciler) Reconcile(ctx context.Context, req ctr
// Use the git operations timeout for the repo.
fetchCtx, cancel := context.WithTimeout(ctx, origin.Spec.Timeout.Duration)
defer cancel()
if err := fetch(fetchCtx, tmp, pushBranch, access); err != nil && err != errRemoteBranchMissing {
return failWithError(err)
}
if err = switchBranch(repo, pushBranch); err != nil {
if err := switchToBranch(repo, fetchCtx, pushBranch, access); err != nil && err != errRemoteBranchMissing {
return failWithError(err)
}
}
Expand Down Expand Up @@ -556,25 +553,6 @@ func cloneInto(ctx context.Context, access repoAccess, ref *sourcev1.GitReposito
return libgit2.OpenRepository(path)
}

// switchBranch switches the repo from the current branch to the
// branch given. If the branch does not exist, it is created using the
// head as the starting point.
func switchBranch(repo *libgit2.Repository, pushBranch string) error {
if err := repo.SetHead(fmt.Sprintf("refs/heads/%s", pushBranch)); err != nil {
head, err := headCommit(repo)
if err != nil {
return err
}
defer head.Free()

branch, err := repo.CreateBranch(pushBranch, head, false)
defer branch.Free()
return err
}

return nil
}

func headCommit(repo *libgit2.Repository) (*libgit2.Commit, error) {
head, err := repo.Head()
if err != nil {
Expand Down Expand Up @@ -748,35 +726,65 @@ func (r *ImageUpdateAutomationReconciler) getSigningEntity(ctx context.Context,

var errRemoteBranchMissing = errors.New("remote branch missing")

// fetch gets the remote branch given and updates the local branch
// head of the same name, so it can be switched to. If the fetch
// completes, it returns nil; if the remote branch is missing, it
// returns errRemoteBranchMissing (this is to work in sympathy with
// `switchBranch`, which will create the branch if it doesn't
// exist). For any other problem it will return the error.
func fetch(ctx context.Context, path string, branch string, access repoAccess) error {
refspec := fmt.Sprintf("refs/heads/%s:refs/heads/%s", branch, branch)
repo, err := libgit2.OpenRepository(path)
if err != nil {
// switchToBranch switches to a branch after fetching latest from upstream.
// If the branch does not exist, it is created using the head as the starting point.
func switchToBranch(repo *libgit2.Repository, ctx context.Context, branch string, access repoAccess) error {
checkoutOpts := &libgit2.CheckoutOpts{
// the remote branch should take precedence if it exists at this point in time.
Strategy: libgit2.CheckoutForce,
}

branchRef := fmt.Sprintf("origin/%s", branch)
remoteBranch, err := repo.LookupBranch(branchRef, libgit2.BranchRemote)
if err != nil && !libgit2.IsErrorCode(err, libgit2.ErrorCodeNotFound) {
return err
}
defer repo.Free()
origin, err := repo.Remotes.Lookup(originRemote)
if remoteBranch != nil {
defer remoteBranch.Free()
}
err = nil

var commit *libgit2.Commit
// tries to get tip commit from remote branch, if it exists.
// otherwise gets the commit that local head is pointing to.
if remoteBranch != nil {
commit, err = repo.LookupCommit(remoteBranch.Target())
} else {
head, err := repo.Head()
if err != nil {
return fmt.Errorf("cannot get repo head: %w", err)
}
commit, err = repo.LookupCommit(head.Target())
}
if err != nil {
return err
return fmt.Errorf("cannot find the head commit: %w", err)
}
defer origin.Free()
err = origin.Fetch(
[]string{refspec},
&libgit2.FetchOptions{
RemoteCallbacks: access.remoteCallbacks(ctx),
ProxyOptions: libgit2.ProxyOptions{Type: libgit2.ProxyTypeAuto},
}, "",
)
if err != nil && libgit2.IsErrorCode(err, libgit2.ErrorCodeNotFound) {
return errRemoteBranchMissing
defer commit.Free()

localBranch, err := repo.LookupBranch(branch, libgit2.BranchLocal)
if err != nil && !libgit2.IsErrorCode(err, libgit2.ErrorCodeNotFound) {
return fmt.Errorf("cannot lookup branch '%s': %w", branch, err)
}
if localBranch == nil {
localBranch, err = repo.CreateBranch(branch, commit, false)
}
return err
if localBranch == nil {
return fmt.Errorf("cannot create local branch '%s': %w", branch, err)
}
defer localBranch.Free()

tree, err := repo.LookupTree(commit.TreeId())
if err != nil {
return fmt.Errorf("cannot lookup tree for branch '%s': %w", branch, err)
}
defer tree.Free()

err = repo.CheckoutTree(tree, checkoutOpts)
if err != nil {
return fmt.Errorf("cannot checkout tree for branch '%s': %w", branch, err)
}

return repo.SetHead("refs/heads/" + branch)
}

// push pushes the branch given to the origin using the git library
Expand Down
34 changes: 34 additions & 0 deletions controllers/update_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -990,6 +990,10 @@ Images:
})

It("creates and pushes the push branch", func() {
initialHead, err := headFromBranch(localRepo, branch)
Expect(err).ToNot(HaveOccurred())
defer initialHead.Free()

// pull the head commit we just pushed, so it's not
// considered a new commit when checking for a commit
// made by automation.
Expand All @@ -1003,9 +1007,19 @@ Images:
Expect(err).ToNot(HaveOccurred())
defer commit.Free()
Expect(commit.Message()).To(Equal(commitMessage))

// previous commits should still exist in the tree.
// regression check to ensure previous commits were not squashed.
oldCommit, err := localRepo.LookupCommit(initialHead.Id())
Expect(err).ToNot(HaveOccurred())
Expect(oldCommit).ToNot(BeNil())
})

It("pushes another commit to the existing push branch", func() {
initialHead, err := headFromBranch(localRepo, branch)
Expect(err).ToNot(HaveOccurred())
defer initialHead.Free()

// pull the head commit we just pushed, so it's not
// considered a new commit when checking for a commit
// made by automation.
Expand All @@ -1027,9 +1041,19 @@ Images:
head, err = getRemoteHead(localRepo, pushBranch)
Expect(err).NotTo(HaveOccurred())
Expect(head.String()).NotTo(Equal(headHash))

// previous commits should still exist in the tree.
// regression check to ensure previous commits were not squashed.
oldCommit, err := localRepo.LookupCommit(initialHead.Id())
Expect(err).ToNot(HaveOccurred())
Expect(oldCommit).ToNot(BeNil())
})

It("still pushes to the push branch after it's merged", func() {
initialHead, err := headFromBranch(localRepo, branch)
Expect(err).ToNot(HaveOccurred())
defer initialHead.Free()

preChangeCommitId := commitIdFromBranch(localRepo, branch)

// observe the first commit
Expand Down Expand Up @@ -1057,6 +1081,12 @@ Images:
head, err = getRemoteHead(localRepo, pushBranch)
Expect(err).NotTo(HaveOccurred())
Expect(head.String()).NotTo(Equal(headHash))

// previous commits should still exist in the tree.
// regression check to ensure previous commits were not squashed.
oldCommit, err := localRepo.LookupCommit(initialHead.Id())
Expect(err).ToNot(HaveOccurred())
Expect(oldCommit).ToNot(BeNil())
})

AfterEach(func() {
Expand Down Expand Up @@ -1731,6 +1761,10 @@ func rebase(repo *git2go.Repository, sourceBranch, targetBranch string) (*git2go

// Check operation index is correct
rebaseOperationIndex, err = rebase.CurrentOperationIndex()
if err != nil {
return nil, err
}

if int(rebaseOperationIndex) != op {
return nil, errors.New("Bad operation index")
}
Expand Down

0 comments on commit 4b35fe3

Please sign in to comment.