Skip to content

Commit

Permalink
Update image-autobumper to correctly utilize go-git (kyma-project#12171)
Browse files Browse the repository at this point in the history
* Update image-autobumper to correctly utilize go-git

* Add explanation for the refspec format
  • Loading branch information
KacperMalachowski committed Nov 6, 2024
1 parent b7a311b commit 1c99031
Showing 1 changed file with 51 additions and 42 deletions.
93 changes: 51 additions & 42 deletions pkg/github/bumper/bumper.go
Original file line number Diff line number Diff line change
Expand Up @@ -101,51 +101,59 @@ func gitCommit(commitMsg string, committerName, committerEmail string, workTree
return nil
}

func gitPush(remote, remoteBranch string, repo *git.Repository, auth transport.AuthMethod, dryrun bool) error {
if _, err := repo.CreateRemote(&config.RemoteConfig{
Name: forkRemoteName,
URLs: []string{remote},
}); err != nil && err != git.ErrRemoteExists {
return fmt.Errorf("create remote: %w", err)
}

remoteRef, err := repo.Reference(plumbing.ReferenceName(fmt.Sprintf("refs/remotes/%s/%s", forkRemoteName, remoteBranch)), true)
if err != nil && err != plumbing.ErrReferenceNotFound {
return fmt.Errorf("get remote ref: %w", err)
func gitPush(remote, remoteBranch, baseBranch string, repo *git.Repository, auth transport.AuthMethod, dryrun bool) error {
// Add the remote if it doesn't exist with the remote url
_, err := repo.Remote(forkRemoteName)
if err != nil {
_, err := repo.CreateRemote(&config.RemoteConfig{
Name: forkRemoteName,
URLs: []string{remote},
})
if err != nil {
return fmt.Errorf("create remote: %w", err)
}
}

remoteTreeRef := ""
if remoteRef != nil {
remoteTreeRef = remoteRef.Hash().String()
// Get the remote head commit
remoteRef, err := repo.Reference(plumbing.NewRemoteReferenceName(forkRemoteName, remoteBranch), true)
if err != nil {
return fmt.Errorf("get remote reference: %w", err)
}

localRef, err := repo.Head()
// Get the local head commit
localRef, err := repo.Reference(plumbing.NewBranchReferenceName(baseBranch), true)
if err != nil {
return fmt.Errorf("get local ref: %w", err)
return fmt.Errorf("get local reference: %w", err)
}

localTreeRef := localRef.Hash().String()
// Check if the remote head commit is the same as the local head commit
if remoteRef.Hash() == localRef.Hash() {
logrus.Info("Remote is up to date, quitting.")
return nil
}

if dryrun {
logrus.Info("[Dryrun] Skip git push with: ")
logrus.Info(forkRemoteName, remoteBranch)
logrus.Infof("[Dryrun] Pushing to %s %s", remote, remoteRef.Name())
return nil
}
// Avoid doing metadata-only pushes that re-trigger tests and remove lgtm
if localTreeRef != remoteTreeRef {
logrus.Info("Pushing to remote...")
if err := repo.Push(&git.PushOptions{
Force: true,
RemoteName: forkRemoteName,
RefSpecs: []config.RefSpec{config.RefSpec(fmt.Sprintf("HEAD:%s", remoteBranch))},
Auth: auth,
// Do not fail if the remote branch is already up-to-date
}); err != nil && err != git.NoErrAlreadyUpToDate {
return fmt.Errorf("push to remote: %w", err)
}
} else {
logrus.Info("Not pushing as up-to-date remote branch already exists")

// Push the changes from local main to the remote.
// We need to use the + sign to force push the changes.
// We need refs/heads/* on the remote branch correctly push the branch using go-git.
// See: https://github.com/go-git/go-git/issues/712#issuecomment-1467085888
refSpecString := fmt.Sprintf("+%s:refs/heads/%s", localRef.Name(), remoteBranch)
logrus.Infof("Pushing changes using %s refspec", refSpecString)
err = repo.Push(&git.PushOptions{
RemoteName: forkRemoteName,
RefSpecs: []config.RefSpec{
config.RefSpec(refSpecString),
},
Auth: auth,
})
if err != nil {
return fmt.Errorf("push changes to the remote branch: %w", err)
}

return nil
}

Expand Down Expand Up @@ -266,6 +274,14 @@ func processGitHub(o *Options, prh PRHandler) error {
}
}

if o.GitHubBaseBranch == "" {
repo, err := gc.GetRepo(o.GitHubOrg, o.GitHubRepo)
if err != nil {
return fmt.Errorf("detect default remote branch for %s/%s: %w", o.GitHubOrg, o.GitHubRepo, err)
}
o.GitHubBaseBranch = repo.DefaultBranch
}

gitRepo, err := git.PlainOpen(o.RepoPath)
if err != nil {
return fmt.Errorf("open git repo: %w", err)
Expand Down Expand Up @@ -302,26 +318,19 @@ func processGitHub(o *Options, prh PRHandler) error {
}
}

remote := fmt.Sprintf("https://%s:%s@%s/%s/%s.git", o.GitHubLogin, string(secret.GetTokenGenerator(o.GitHubToken)()), githubHost, o.GitHubLogin, o.RemoteName)
remote := fmt.Sprintf("https://%s/%s/%s.git", githubHost, o.GitHubLogin, o.RemoteName)
authMethod := &http.BasicAuth{
Username: o.GitHubLogin,
Password: string(secret.GetTokenGenerator(o.GitHubToken)()),
}
if err := gitPush(remote, o.HeadBranchName, gitRepo, authMethod, o.SkipPullRequest); err != nil {
if err := gitPush(remote, o.HeadBranchName, o.GitHubBaseBranch, gitRepo, authMethod, o.SkipPullRequest); err != nil {
return fmt.Errorf("push changes to the remote branch: %w", err)
}

summary, body, err := prh.PRTitleBody()
if err != nil {
return fmt.Errorf("creating PR summary and body: %w", err)
}
if o.GitHubBaseBranch == "" {
repo, err := gc.GetRepo(o.GitHubOrg, o.GitHubRepo)
if err != nil {
return fmt.Errorf("detect default remote branch for %s/%s: %w", o.GitHubOrg, o.GitHubRepo, err)
}
o.GitHubBaseBranch = repo.DefaultBranch
}
if err := updatePRWithLabels(gc, o.GitHubOrg, o.GitHubRepo, getAssignment(o.AssignTo), o.GitHubLogin, o.GitHubBaseBranch, o.HeadBranchName, updater.PreventMods, summary, body, o.Labels, o.SkipPullRequest); err != nil {
return fmt.Errorf("to create the PR: %w", err)
}
Expand Down

0 comments on commit 1c99031

Please sign in to comment.