You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

merge_squash.go 3.0KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  1. // Copyright 2023 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package pull
  4. import (
  5. "fmt"
  6. repo_model "code.gitea.io/gitea/models/repo"
  7. user_model "code.gitea.io/gitea/models/user"
  8. "code.gitea.io/gitea/modules/container"
  9. "code.gitea.io/gitea/modules/git"
  10. "code.gitea.io/gitea/modules/log"
  11. "code.gitea.io/gitea/modules/setting"
  12. )
  13. // doMergeStyleSquash gets a commit author signature for squash commits
  14. func getAuthorSignatureSquash(ctx *mergeContext) (*git.Signature, error) {
  15. if err := ctx.pr.Issue.LoadPoster(ctx); err != nil {
  16. log.Error("%-v Issue[%d].LoadPoster: %v", ctx.pr, ctx.pr.Issue.ID, err)
  17. return nil, err
  18. }
  19. // Try to get an signature from the same user in one of the commits, as the
  20. // poster email might be private or commits might have a different signature
  21. // than the primary email address of the poster.
  22. gitRepo, closer, err := git.RepositoryFromContextOrOpen(ctx, ctx.tmpBasePath)
  23. if err != nil {
  24. log.Error("%-v Unable to open base repository: %v", ctx.pr, err)
  25. return nil, err
  26. }
  27. defer closer.Close()
  28. commits, err := gitRepo.CommitsBetweenIDs(trackingBranch, "HEAD")
  29. if err != nil {
  30. log.Error("%-v Unable to get commits between: %s %s: %v", ctx.pr, "HEAD", trackingBranch, err)
  31. return nil, err
  32. }
  33. uniqueEmails := make(container.Set[string])
  34. for _, commit := range commits {
  35. if commit.Author != nil && uniqueEmails.Add(commit.Author.Email) {
  36. commitUser, _ := user_model.GetUserByEmail(ctx, commit.Author.Email)
  37. if commitUser != nil && commitUser.ID == ctx.pr.Issue.Poster.ID {
  38. return commit.Author, nil
  39. }
  40. }
  41. }
  42. return ctx.pr.Issue.Poster.NewGitSig(), nil
  43. }
  44. // doMergeStyleSquash squashes the tracking branch on the current HEAD (=base)
  45. func doMergeStyleSquash(ctx *mergeContext, message string) error {
  46. sig, err := getAuthorSignatureSquash(ctx)
  47. if err != nil {
  48. return fmt.Errorf("getAuthorSignatureSquash: %w", err)
  49. }
  50. cmdMerge := git.NewCommand(ctx, "merge", "--squash").AddDynamicArguments(trackingBranch)
  51. if err := runMergeCommand(ctx, repo_model.MergeStyleSquash, cmdMerge); err != nil {
  52. log.Error("%-v Unable to merge --squash tracking into base: %v", ctx.pr, err)
  53. return err
  54. }
  55. if setting.Repository.PullRequest.AddCoCommitterTrailers && ctx.committer.String() != sig.String() {
  56. // add trailer
  57. message += fmt.Sprintf("\nCo-authored-by: %s\nCo-committed-by: %s\n", sig.String(), sig.String())
  58. }
  59. cmdCommit := git.NewCommand(ctx, "commit").
  60. AddOptionFormat("--author='%s <%s>'", sig.Name, sig.Email).
  61. AddOptionFormat("--message=%s", message)
  62. if ctx.signKeyID == "" {
  63. cmdCommit.AddArguments("--no-gpg-sign")
  64. } else {
  65. cmdCommit.AddOptionFormat("-S%s", ctx.signKeyID)
  66. }
  67. if err := cmdCommit.Run(ctx.RunOpts()); err != nil {
  68. log.Error("git commit %-v: %v\n%s\n%s", ctx.pr, err, ctx.outbuf.String(), ctx.errbuf.String())
  69. return fmt.Errorf("git commit [%s:%s -> %s:%s]: %w\n%s\n%s", ctx.pr.HeadRepo.FullName(), ctx.pr.HeadBranch, ctx.pr.BaseRepo.FullName(), ctx.pr.BaseBranch, err, ctx.outbuf.String(), ctx.errbuf.String())
  70. }
  71. ctx.outbuf.Reset()
  72. ctx.errbuf.Reset()
  73. return nil
  74. }