Skip to content

Commit eb792d9

Browse files
lunnyearl-warren
authored andcommitted
Move database operations of merging a pull request to post receive hook and add a transaction (#30805)
Merging PR may fail because of various problems. The pull request may have a dirty state because there is no transaction when merging a pull request. ref go-gitea/gitea#25741 (comment) This PR moves all database update operations to post-receive handler for merging a pull request and having a database transaction. That means if database operations fail, then the git merging will fail, the git client will get a fail result. There are already many tests for pull request merging, so we don't need to add a new one. --------- Co-authored-by: wxiaoguang <wxiaoguang@gmail.com> (cherry picked from commit ebf0c969403d91ed80745ff5bd7dfbdb08174fc7) Conflicts: modules/private/hook.go routers/private/hook_post_receive.go trivial conflicts because 263a716cb5 * Performance optimization for git push (#30104) was not cherry-picked and because of 998a431 Do not update PRs based on events that happened before they existed
1 parent 1f56a49 commit eb792d9

8 files changed

Lines changed: 150 additions & 18 deletions

File tree

cmd/hook.go

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -366,6 +366,7 @@ Forgejo or set your environment appropriately.`, "")
366366
isWiki, _ := strconv.ParseBool(os.Getenv(repo_module.EnvRepoIsWiki))
367367
repoName := os.Getenv(repo_module.EnvRepoName)
368368
pusherID, _ := strconv.ParseInt(os.Getenv(repo_module.EnvPusherID), 10, 64)
369+
prID, _ := strconv.ParseInt(os.Getenv(repo_module.EnvPRID), 10, 64)
369370
pusherName := os.Getenv(repo_module.EnvPusherName)
370371

371372
hookOptions := private.HookOptions{
@@ -375,6 +376,8 @@ Forgejo or set your environment appropriately.`, "")
375376
GitObjectDirectory: os.Getenv(private.GitObjectDirectory),
376377
GitQuarantinePath: os.Getenv(private.GitQuarantinePath),
377378
GitPushOptions: pushOptions(),
379+
PullRequestID: prID,
380+
PushTrigger: repo_module.PushTrigger(os.Getenv(repo_module.EnvPushTrigger)),
378381
}
379382
oldCommitIDs := make([]string, hookBatchSize)
380383
newCommitIDs := make([]string, hookBatchSize)

modules/private/hook.go

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ import (
1111
"time"
1212

1313
"code.gitea.io/gitea/modules/git"
14+
"code.gitea.io/gitea/modules/repository"
1415
"code.gitea.io/gitea/modules/setting"
1516
)
1617

@@ -53,6 +54,7 @@ type HookOptions struct {
5354
GitQuarantinePath string
5455
GitPushOptions GitPushOptions
5556
PullRequestID int64
57+
PushTrigger repository.PushTrigger
5658
DeployKeyID int64 // if the pusher is a DeployKey, then UserID is the repo's org user.
5759
IsWiki bool
5860
ActionPerm int

modules/repository/env.go

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -25,11 +25,19 @@ const (
2525
EnvKeyID = "GITEA_KEY_ID" // public key ID
2626
EnvDeployKeyID = "GITEA_DEPLOY_KEY_ID"
2727
EnvPRID = "GITEA_PR_ID"
28+
EnvPushTrigger = "GITEA_PUSH_TRIGGER"
2829
EnvIsInternal = "GITEA_INTERNAL_PUSH"
2930
EnvAppURL = "GITEA_ROOT_URL"
3031
EnvActionPerm = "GITEA_ACTION_PERM"
3132
)
3233

34+
type PushTrigger string
35+
36+
const (
37+
PushTriggerPRMergeToBase PushTrigger = "pr-merge-to-base"
38+
PushTriggerPRUpdateWithBase PushTrigger = "pr-update-with-base"
39+
)
40+
3341
// InternalPushingEnvironment returns an os environment to switch off hooks on push
3442
// It is recommended to avoid using this unless you are pushing within a transaction
3543
// or if you absolutely are sure that post-receive and pre-receive will do nothing

routers/private/hook_post_receive.go

Lines changed: 63 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -4,20 +4,26 @@
44
package private
55

66
import (
7+
"context"
78
"fmt"
89
"net/http"
910
"strconv"
1011
"time"
1112

13+
"code.gitea.io/gitea/models/db"
1214
git_model "code.gitea.io/gitea/models/git"
1315
issues_model "code.gitea.io/gitea/models/issues"
16+
pull_model "code.gitea.io/gitea/models/pull"
1417
repo_model "code.gitea.io/gitea/models/repo"
18+
user_model "code.gitea.io/gitea/models/user"
19+
"code.gitea.io/gitea/modules/cache"
1520
"code.gitea.io/gitea/modules/git"
1621
"code.gitea.io/gitea/modules/gitrepo"
1722
"code.gitea.io/gitea/modules/log"
1823
"code.gitea.io/gitea/modules/private"
1924
repo_module "code.gitea.io/gitea/modules/repository"
2025
"code.gitea.io/gitea/modules/setting"
26+
timeutil "code.gitea.io/gitea/modules/timeutil"
2127
"code.gitea.io/gitea/modules/util"
2228
"code.gitea.io/gitea/modules/web"
2329
gitea_context "code.gitea.io/gitea/services/context"
@@ -155,6 +161,14 @@ func HookPostReceive(ctx *gitea_context.PrivateContext) {
155161
}
156162
}
157163

164+
// handle pull request merging, a pull request action should push at least 1 commit
165+
if opts.PushTrigger == repo_module.PushTriggerPRMergeToBase {
166+
handlePullRequestMerging(ctx, opts, ownerName, repoName, updates)
167+
if ctx.Written() {
168+
return
169+
}
170+
}
171+
158172
// Handle Push Options
159173
if len(opts.GitPushOptions) > 0 {
160174
// load the repository
@@ -302,3 +316,52 @@ func HookPostReceive(ctx *gitea_context.PrivateContext) {
302316
RepoWasEmpty: wasEmpty,
303317
})
304318
}
319+
320+
func loadContextCacheUser(ctx context.Context, id int64) (*user_model.User, error) {
321+
return cache.GetWithContextCache(ctx, "hook_post_receive_user", id, func() (*user_model.User, error) {
322+
return user_model.GetUserByID(ctx, id)
323+
})
324+
}
325+
326+
// handlePullRequestMerging handle pull request merging, a pull request action should push at least 1 commit
327+
func handlePullRequestMerging(ctx *gitea_context.PrivateContext, opts *private.HookOptions, ownerName, repoName string, updates []*repo_module.PushUpdateOptions) {
328+
if len(updates) == 0 {
329+
ctx.JSON(http.StatusInternalServerError, private.HookPostReceiveResult{
330+
Err: fmt.Sprintf("Pushing a merged PR (pr:%d) no commits pushed ", opts.PullRequestID),
331+
})
332+
return
333+
}
334+
335+
pr, err := issues_model.GetPullRequestByID(ctx, opts.PullRequestID)
336+
if err != nil {
337+
log.Error("GetPullRequestByID[%d]: %v", opts.PullRequestID, err)
338+
ctx.JSON(http.StatusInternalServerError, private.HookPostReceiveResult{Err: "GetPullRequestByID failed"})
339+
return
340+
}
341+
342+
pusher, err := loadContextCacheUser(ctx, opts.UserID)
343+
if err != nil {
344+
log.Error("Failed to Update: %s/%s Error: %v", ownerName, repoName, err)
345+
ctx.JSON(http.StatusInternalServerError, private.HookPostReceiveResult{Err: "Load pusher user failed"})
346+
return
347+
}
348+
349+
pr.MergedCommitID = updates[len(updates)-1].NewCommitID
350+
pr.MergedUnix = timeutil.TimeStampNow()
351+
pr.Merger = pusher
352+
pr.MergerID = pusher.ID
353+
err = db.WithTx(ctx, func(ctx context.Context) error {
354+
// Removing an auto merge pull and ignore if not exist
355+
if err := pull_model.DeleteScheduledAutoMerge(ctx, pr.ID); err != nil && !db.IsErrNotExist(err) {
356+
return fmt.Errorf("DeleteScheduledAutoMerge[%d]: %v", opts.PullRequestID, err)
357+
}
358+
if _, err := pr.SetMerged(ctx); err != nil {
359+
return fmt.Errorf("SetMerged failed: %s/%s Error: %v", ownerName, repoName, err)
360+
}
361+
return nil
362+
})
363+
if err != nil {
364+
log.Error("Failed to update PR to merged: %v", err)
365+
ctx.JSON(http.StatusInternalServerError, private.HookPostReceiveResult{Err: "Failed to update PR to merged"})
366+
}
367+
}
Lines changed: 49 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
// Copyright 2024 The Gitea Authors. All rights reserved.
2+
// SPDX-License-Identifier: MIT
3+
4+
package private
5+
6+
import (
7+
"testing"
8+
9+
"code.gitea.io/gitea/models/db"
10+
issues_model "code.gitea.io/gitea/models/issues"
11+
pull_model "code.gitea.io/gitea/models/pull"
12+
repo_model "code.gitea.io/gitea/models/repo"
13+
"code.gitea.io/gitea/models/unittest"
14+
user_model "code.gitea.io/gitea/models/user"
15+
"code.gitea.io/gitea/modules/private"
16+
repo_module "code.gitea.io/gitea/modules/repository"
17+
"code.gitea.io/gitea/services/contexttest"
18+
19+
"github.com/stretchr/testify/assert"
20+
)
21+
22+
func TestHandlePullRequestMerging(t *testing.T) {
23+
assert.NoError(t, unittest.PrepareTestDatabase())
24+
pr, err := issues_model.GetUnmergedPullRequest(db.DefaultContext, 1, 1, "branch2", "master", issues_model.PullRequestFlowGithub)
25+
assert.NoError(t, err)
26+
assert.NoError(t, pr.LoadBaseRepo(db.DefaultContext))
27+
28+
user1 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 1})
29+
30+
err = pull_model.ScheduleAutoMerge(db.DefaultContext, user1, pr.ID, repo_model.MergeStyleSquash, "squash merge a pr")
31+
assert.NoError(t, err)
32+
33+
autoMerge := unittest.AssertExistsAndLoadBean(t, &pull_model.AutoMerge{PullID: pr.ID})
34+
35+
ctx, resp := contexttest.MockPrivateContext(t, "/")
36+
handlePullRequestMerging(ctx, &private.HookOptions{
37+
PullRequestID: pr.ID,
38+
UserID: 2,
39+
}, pr.BaseRepo.OwnerName, pr.BaseRepo.Name, []*repo_module.PushUpdateOptions{
40+
{NewCommitID: "01234567"},
41+
})
42+
assert.Equal(t, 0, len(resp.Body.String()))
43+
pr, err = issues_model.GetPullRequestByID(db.DefaultContext, pr.ID)
44+
assert.NoError(t, err)
45+
assert.True(t, pr.HasMerged)
46+
assert.EqualValues(t, "01234567", pr.MergedCommitID)
47+
48+
unittest.AssertNotExistsBean(t, &pull_model.AutoMerge{ID: autoMerge.ID})
49+
}

services/contexttest/context_tests.go

Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,19 @@ func MockAPIContext(t *testing.T, reqPath string) (*context.APIContext, *httptes
8686
return ctx, resp
8787
}
8888

89+
func MockPrivateContext(t *testing.T, reqPath string) (*context.PrivateContext, *httptest.ResponseRecorder) {
90+
resp := httptest.NewRecorder()
91+
req := mockRequest(t, reqPath)
92+
base, baseCleanUp := context.NewBaseContext(resp, req)
93+
base.Data = middleware.GetContextData(req.Context())
94+
base.Locale = &translation.MockLocale{}
95+
ctx := &context.PrivateContext{Base: base}
96+
_ = baseCleanUp // during test, it doesn't need to do clean up. TODO: this can be improved later
97+
chiCtx := chi.NewRouteContext()
98+
ctx.Base.AppendContextValue(chi.RouteCtxKey, chiCtx)
99+
return ctx, resp
100+
}
101+
89102
// LoadRepo load a repo into a test context.
90103
func LoadRepo(t *testing.T, ctx gocontext.Context, repoID int64) {
91104
var doer *user_model.User

services/pull/merge.go

Lines changed: 10 additions & 17 deletions
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,6 @@ import (
1818
git_model "code.gitea.io/gitea/models/git"
1919
issues_model "code.gitea.io/gitea/models/issues"
2020
access_model "code.gitea.io/gitea/models/perm/access"
21-
pull_model "code.gitea.io/gitea/models/pull"
2221
repo_model "code.gitea.io/gitea/models/repo"
2322
"code.gitea.io/gitea/models/unit"
2423
user_model "code.gitea.io/gitea/models/user"
@@ -168,12 +167,6 @@ func Merge(ctx context.Context, pr *issues_model.PullRequest, doer *user_model.U
168167
pullWorkingPool.CheckIn(fmt.Sprint(pr.ID))
169168
defer pullWorkingPool.CheckOut(fmt.Sprint(pr.ID))
170169

171-
// Removing an auto merge pull and ignore if not exist
172-
// FIXME: is this the correct point to do this? Shouldn't this be after IsMergeStyleAllowed?
173-
if err := pull_model.DeleteScheduledAutoMerge(ctx, pr.ID); err != nil && !db.IsErrNotExist(err) {
174-
return err
175-
}
176-
177170
prUnit, err := pr.BaseRepo.GetUnit(ctx, unit.TypePullRequests)
178171
if err != nil {
179172
log.Error("pr.BaseRepo.GetUnit(unit.TypePullRequests): %v", err)
@@ -190,17 +183,15 @@ func Merge(ctx context.Context, pr *issues_model.PullRequest, doer *user_model.U
190183
AddTestPullRequestTask(ctx, doer, pr.BaseRepo.ID, pr.BaseBranch, false, "", "", 0)
191184
}()
192185

193-
pr.MergedCommitID, err = doMergeAndPush(ctx, pr, doer, mergeStyle, expectedHeadCommitID, message)
186+
_, err = doMergeAndPush(ctx, pr, doer, mergeStyle, expectedHeadCommitID, message, repo_module.PushTriggerPRMergeToBase)
194187
if err != nil {
195188
return err
196189
}
197190

198-
pr.MergedUnix = timeutil.TimeStampNow()
199-
pr.Merger = doer
200-
pr.MergerID = doer.ID
201-
202-
if _, err := pr.SetMerged(ctx); err != nil {
203-
log.Error("SetMerged %-v: %v", pr, err)
191+
// reload pull request because it has been updated by post receive hook
192+
pr, err = issues_model.GetPullRequestByID(ctx, pr.ID)
193+
if err != nil {
194+
return err
204195
}
205196

206197
if err := pr.LoadIssue(ctx); err != nil {
@@ -251,7 +242,7 @@ func Merge(ctx context.Context, pr *issues_model.PullRequest, doer *user_model.U
251242
}
252243

253244
// doMergeAndPush performs the merge operation without changing any pull information in database and pushes it up to the base repository
254-
func doMergeAndPush(ctx context.Context, pr *issues_model.PullRequest, doer *user_model.User, mergeStyle repo_model.MergeStyle, expectedHeadCommitID, message string) (string, error) {
245+
func doMergeAndPush(ctx context.Context, pr *issues_model.PullRequest, doer *user_model.User, mergeStyle repo_model.MergeStyle, expectedHeadCommitID, message string, pushTrigger repo_module.PushTrigger) (string, error) {
255246
// Clone base repo.
256247
mergeCtx, cancel, err := createTemporaryRepoForMerge(ctx, pr, doer, expectedHeadCommitID)
257248
if err != nil {
@@ -324,11 +315,13 @@ func doMergeAndPush(ctx context.Context, pr *issues_model.PullRequest, doer *use
324315
pr.BaseRepo.Name,
325316
pr.ID,
326317
)
318+
319+
mergeCtx.env = append(mergeCtx.env, repo_module.EnvPushTrigger+"="+string(pushTrigger))
327320
pushCmd := git.NewCommand(ctx, "push", "origin").AddDynamicArguments(baseBranch + ":" + git.BranchPrefix + pr.BaseBranch)
328321

329322
// Push back to upstream.
330-
// TODO: this cause an api call to "/api/internal/hook/post-receive/...",
331-
// that prevents us from doint the whole merge in one db transaction
323+
// This cause an api call to "/api/internal/hook/post-receive/...",
324+
// If it's merge, all db transaction and operations should be there but not here to prevent deadlock.
332325
if err := pushCmd.Run(mergeCtx.RunOpts()); err != nil {
333326
if strings.Contains(mergeCtx.errbuf.String(), "non-fast-forward") {
334327
return "", &git.ErrPushOutOfDate{

services/pull/update.go

Lines changed: 2 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ import (
1515
user_model "code.gitea.io/gitea/models/user"
1616
"code.gitea.io/gitea/modules/git"
1717
"code.gitea.io/gitea/modules/log"
18+
"code.gitea.io/gitea/modules/repository"
1819
)
1920

2021
// Update updates pull request with base branch.
@@ -72,7 +73,7 @@ func Update(ctx context.Context, pr *issues_model.PullRequest, doer *user_model.
7273
BaseBranch: pr.HeadBranch,
7374
}
7475

75-
_, err = doMergeAndPush(ctx, reversePR, doer, repo_model.MergeStyleMerge, "", message)
76+
_, err = doMergeAndPush(ctx, reversePR, doer, repo_model.MergeStyleMerge, "", message, repository.PushTriggerPRUpdateWithBase)
7677

7778
defer func() {
7879
AddTestPullRequestTask(ctx, doer, reversePR.HeadRepo.ID, reversePR.HeadBranch, false, "", "", 0)

0 commit comments

Comments
 (0)