Skip a HookEventPullRequestSync event if it has the same CommitSHA as an existing HookEventPullRequest event in the ActionRun table. A HookEventPullRequestSync event must only create an ActionRun if the CommitSHA is different from what it was when the PR was open. This guards against a race that can happen when the following is done in parallel: * A commit C is pushed to a repo on branch B * A pull request with head on branch B it is then possible that the pull request is created first, successfully. The commit that was just pushed is not known yet but the PR only references the repository and the B branch so it is fine. A HookEventPullRequest event is sent to the notification queue but not processed immediately. The commit C is pushed and processed successfully. Since the PR already exists and has a head that matches the branch, the head of the PR is updated with the commit C and a HookEventPullRequestSync event is sent to the notification queue. The HookEventPullRequest event is processed and since the head of the PR was updated to be commit C, an ActionRun with CommitSHA C is created. The HookEventPullRequestSync event is then processed and also has a CommitSHA equal to C. Refs: https://codeberg.org/forgejo/forgejo/issues/2009 Reviewed-on: https://codeberg.org/forgejo/forgejo/pulls/2314 Co-authored-by: Earl Warren <contact@earl-warren.org> Co-committed-by: Earl Warren <contact@earl-warren.org>
		
			
				
	
	
		
			50 lines
		
	
	
	
		
			1.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			50 lines
		
	
	
	
		
			1.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2024 The Forgejo Authors
 | 
						|
// SPDX-License-Identifier: MIT
 | 
						|
 | 
						|
package actions
 | 
						|
 | 
						|
import (
 | 
						|
	"testing"
 | 
						|
 | 
						|
	actions_model "code.gitea.io/gitea/models/actions"
 | 
						|
	"code.gitea.io/gitea/models/db"
 | 
						|
	"code.gitea.io/gitea/models/unittest"
 | 
						|
	webhook_module "code.gitea.io/gitea/modules/webhook"
 | 
						|
 | 
						|
	"github.com/stretchr/testify/assert"
 | 
						|
)
 | 
						|
 | 
						|
func Test_SkipPullRequestEvent(t *testing.T) {
 | 
						|
	assert.NoError(t, unittest.PrepareTestDatabase())
 | 
						|
 | 
						|
	repoID := int64(1)
 | 
						|
	commitSHA := "1234"
 | 
						|
 | 
						|
	// event is not webhook_module.HookEventPullRequestSync, never skip
 | 
						|
	assert.False(t, SkipPullRequestEvent(db.DefaultContext, webhook_module.HookEventPullRequest, repoID, commitSHA))
 | 
						|
 | 
						|
	// event is webhook_module.HookEventPullRequestSync but there is nothing in the ActionRun table, do not skip
 | 
						|
	assert.False(t, SkipPullRequestEvent(db.DefaultContext, webhook_module.HookEventPullRequestSync, repoID, commitSHA))
 | 
						|
 | 
						|
	// there is a webhook_module.HookEventPullRequest event but the SHA is different, do not skip
 | 
						|
	index := int64(1)
 | 
						|
	run := &actions_model.ActionRun{
 | 
						|
		Index:     index,
 | 
						|
		Event:     webhook_module.HookEventPullRequest,
 | 
						|
		RepoID:    repoID,
 | 
						|
		CommitSHA: "othersha",
 | 
						|
	}
 | 
						|
	unittest.AssertSuccessfulInsert(t, run)
 | 
						|
	assert.False(t, SkipPullRequestEvent(db.DefaultContext, webhook_module.HookEventPullRequestSync, repoID, commitSHA))
 | 
						|
 | 
						|
	// there already is a webhook_module.HookEventPullRequest with the same SHA, skip
 | 
						|
	index++
 | 
						|
	run = &actions_model.ActionRun{
 | 
						|
		Index:     index,
 | 
						|
		Event:     webhook_module.HookEventPullRequest,
 | 
						|
		RepoID:    repoID,
 | 
						|
		CommitSHA: commitSHA,
 | 
						|
	}
 | 
						|
	unittest.AssertSuccessfulInsert(t, run)
 | 
						|
	assert.True(t, SkipPullRequestEvent(db.DefaultContext, webhook_module.HookEventPullRequestSync, repoID, commitSHA))
 | 
						|
}
 |