0
0
mirror of https://github.com/go-gitea/gitea.git synced 2026-03-26 22:28:13 +01:00

Fix missing workflow_run notifications when updating jobs from multiple runs (#36997)

This PR fixes `notifyWorkflowJobStatusUpdate` to send
`WorkflowRunStatusUpdate` for each affected workflow run instead of only
the first run in the input job list.
This commit is contained in:
Zettat123 2026-03-26 12:48:04 -06:00 committed by GitHub
parent d5a89805d9
commit 8fdd6d1235
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 90 additions and 2 deletions

View File

@ -40,6 +40,8 @@ func notifyWorkflowJobStatusUpdate(ctx context.Context, jobs []*actions_model.Ac
if len(jobs) == 0 {
return
}
// The input jobs may belong to different runs, so track each affected run.
runs := make(map[int64]*actions_model.ActionRun, len(jobs))
for _, job := range jobs {
if err := job.LoadAttributes(ctx); err != nil {
log.Error("Failed to load job attributes: %v", err)
@ -47,10 +49,13 @@ func notifyWorkflowJobStatusUpdate(ctx context.Context, jobs []*actions_model.Ac
}
CreateCommitStatusForRunJobs(ctx, job.Run, job)
notify_service.WorkflowJobStatusUpdate(ctx, job.Run.Repo, job.Run.TriggerUser, job, nil)
if _, ok := runs[job.RunID]; !ok {
runs[job.RunID] = job.Run
}
}
if job := jobs[0]; job.Run != nil && job.Run.Repo != nil {
notify_service.WorkflowRunStatusUpdate(ctx, job.Run.Repo, job.Run.TriggerUser, job.Run)
for _, run := range runs {
notify_service.WorkflowRunStatusUpdate(ctx, run.Repo, run.TriggerUser, run)
}
}

View File

@ -14,6 +14,7 @@ import (
"testing"
"time"
actions_model "code.gitea.io/gitea/models/actions"
auth_model "code.gitea.io/gitea/models/auth"
"code.gitea.io/gitea/models/perm"
"code.gitea.io/gitea/models/repo"
@ -1146,6 +1147,10 @@ func Test_WebhookWorkflowRun(t *testing.T) {
testWorkflowRunEventsOnCancellingAbandonedRun(t, webhookData, false)
},
},
{
name: "WorkflowRunOnStoppingEndlessTasksForMultipleRuns",
testFunc: testWorkflowRunOnStoppingEndlessTasksForMultipleRuns,
},
}
for _, obj := range testCases {
t.Run(obj.name, func(t *testing.T) {
@ -1576,6 +1581,84 @@ jobs:
assert.Equal(t, "user2/"+repoName, webhookData.payloads[1].Repo.FullName)
}
func testWorkflowRunOnStoppingEndlessTasksForMultipleRuns(t *testing.T, webhookData *workflowRunWebhook) {
defer test.MockVariableValue(&setting.Actions.EndlessTaskTimeout, time.Second)()
user2 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
session := loginUser(t, "user2")
token := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeWriteRepository, auth_model.AccessTokenScopeWriteUser)
repoName := "test-workflow-run-stop-endless-tasks"
testRepo := unittest.AssertExistsAndLoadBean(t, &repo.Repository{ID: createActionsTestRepo(t, token, repoName, false).ID})
testAPICreateWebhookForRepo(t, session, "user2", repoName, webhookData.URL, "workflow_run")
runners := make([]*mockRunner, 2)
for i := range runners {
runners[i] = newMockRunner()
runners[i].registerAsRepoRunner(t, "user2", repoName, fmt.Sprintf("mock-runner-%d", i), []string{"ubuntu-latest"}, false)
}
workflowPath1 := ".gitea/workflows/endless-1.yml"
workflowPath2 := ".gitea/workflows/endless-2.yml"
workflowContent1 := `name: endless-1
on:
push:
paths:
- '.gitea/workflows/endless-1.yml'
jobs:
job-1:
runs-on: ubuntu-latest
steps:
- run: echo 'job-1'
`
workflowContent2 := `name: endless-2
on:
push:
paths:
- '.gitea/workflows/endless-2.yml'
jobs:
job-2:
runs-on: ubuntu-latest
steps:
- run: echo 'job-2'
`
opts1 := getWorkflowCreateFileOptions(user2, testRepo.DefaultBranch, "create "+workflowPath1, workflowContent1)
createWorkflowFile(t, token, "user2", repoName, workflowPath1, opts1)
opts2 := getWorkflowCreateFileOptions(user2, testRepo.DefaultBranch, "create "+workflowPath2, workflowContent2)
createWorkflowFile(t, token, "user2", repoName, workflowPath2, opts2)
task1 := runners[0].fetchTask(t)
task2 := runners[1].fetchTask(t)
_, job1, _ := getTaskAndJobAndRunByTaskID(t, task1.Id)
_, job2, _ := getTaskAndJobAndRunByTaskID(t, task2.Id)
require.NotEqual(t, job1.RunID, job2.RunID)
initialRunEventsLen := len(webhookData.payloads)
time.Sleep(2 * time.Second)
require.NoError(t, actions.StopEndlessTasks(t.Context()))
require.Len(t, webhookData.payloads, initialRunEventsLen+2)
var completedRunIDs []int64
for _, payload := range webhookData.payloads[initialRunEventsLen:] {
assert.Equal(t, "completed", payload.Action)
assert.Equal(t, "completed", payload.WorkflowRun.Status)
completedRunIDs = append(completedRunIDs, payload.WorkflowRun.ID)
}
assert.Len(t, completedRunIDs, 2)
assert.Contains(t, completedRunIDs, job1.RunID)
assert.Contains(t, completedRunIDs, job2.RunID)
run1 := unittest.AssertExistsAndLoadBean(t, &actions_model.ActionRun{ID: job1.RunID})
run2 := unittest.AssertExistsAndLoadBean(t, &actions_model.ActionRun{ID: job2.RunID})
assert.Equal(t, actions_model.StatusFailure, run1.Status)
assert.Equal(t, actions_model.StatusFailure, run2.Status)
}
func testWebhookWorkflowRun(t *testing.T, webhookData *workflowRunWebhook) {
// 1. create a new webhook with special webhook for repo1
user2 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})