mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-24 13:53:42 +09:00 
			
		
		
		
	Fix #24769 Fix #32662 Fix #33260 Depends on https://gitea.com/gitea/act/pulls/124 - https://docs.github.com/en/actions/writing-workflows/workflow-syntax-for-github-actions#concurrency ## ⚠️ BREAKING ⚠️ This PR removes the auto-cancellation feature added by #25716. Users need to manually add `concurrency` to workflows to control concurrent workflows or jobs. --------- Signed-off-by: Zettat123 <zettat123@gmail.com> Co-authored-by: Christopher Homberger <christopher.homberger@web.de> Co-authored-by: wxiaoguang <wxiaoguang@gmail.com>
		
			
				
	
	
		
			137 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			137 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2022 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package actions
 | |
| 
 | |
| import (
 | |
| 	"testing"
 | |
| 
 | |
| 	actions_model "code.gitea.io/gitea/models/actions"
 | |
| 
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| )
 | |
| 
 | |
| func Test_jobStatusResolver_Resolve(t *testing.T) {
 | |
| 	tests := []struct {
 | |
| 		name string
 | |
| 		jobs actions_model.ActionJobList
 | |
| 		want map[int64]actions_model.Status
 | |
| 	}{
 | |
| 		{
 | |
| 			name: "no blocked",
 | |
| 			jobs: actions_model.ActionJobList{
 | |
| 				{ID: 1, JobID: "1", Status: actions_model.StatusWaiting, Needs: []string{}},
 | |
| 				{ID: 2, JobID: "2", Status: actions_model.StatusWaiting, Needs: []string{}},
 | |
| 				{ID: 3, JobID: "3", Status: actions_model.StatusWaiting, Needs: []string{}},
 | |
| 			},
 | |
| 			want: map[int64]actions_model.Status{},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "single blocked",
 | |
| 			jobs: actions_model.ActionJobList{
 | |
| 				{ID: 1, JobID: "1", Status: actions_model.StatusSuccess, Needs: []string{}},
 | |
| 				{ID: 2, JobID: "2", Status: actions_model.StatusBlocked, Needs: []string{"1"}},
 | |
| 				{ID: 3, JobID: "3", Status: actions_model.StatusWaiting, Needs: []string{}},
 | |
| 			},
 | |
| 			want: map[int64]actions_model.Status{
 | |
| 				2: actions_model.StatusWaiting,
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "multiple blocked",
 | |
| 			jobs: actions_model.ActionJobList{
 | |
| 				{ID: 1, JobID: "1", Status: actions_model.StatusSuccess, Needs: []string{}},
 | |
| 				{ID: 2, JobID: "2", Status: actions_model.StatusBlocked, Needs: []string{"1"}},
 | |
| 				{ID: 3, JobID: "3", Status: actions_model.StatusBlocked, Needs: []string{"1"}},
 | |
| 			},
 | |
| 			want: map[int64]actions_model.Status{
 | |
| 				2: actions_model.StatusWaiting,
 | |
| 				3: actions_model.StatusWaiting,
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "chain blocked",
 | |
| 			jobs: actions_model.ActionJobList{
 | |
| 				{ID: 1, JobID: "1", Status: actions_model.StatusFailure, Needs: []string{}},
 | |
| 				{ID: 2, JobID: "2", Status: actions_model.StatusBlocked, Needs: []string{"1"}},
 | |
| 				{ID: 3, JobID: "3", Status: actions_model.StatusBlocked, Needs: []string{"2"}},
 | |
| 			},
 | |
| 			want: map[int64]actions_model.Status{
 | |
| 				2: actions_model.StatusSkipped,
 | |
| 				3: actions_model.StatusSkipped,
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "loop need",
 | |
| 			jobs: actions_model.ActionJobList{
 | |
| 				{ID: 1, JobID: "1", Status: actions_model.StatusBlocked, Needs: []string{"3"}},
 | |
| 				{ID: 2, JobID: "2", Status: actions_model.StatusBlocked, Needs: []string{"1"}},
 | |
| 				{ID: 3, JobID: "3", Status: actions_model.StatusBlocked, Needs: []string{"2"}},
 | |
| 			},
 | |
| 			want: map[int64]actions_model.Status{},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "`if` is not empty and all jobs in `needs` completed successfully",
 | |
| 			jobs: actions_model.ActionJobList{
 | |
| 				{ID: 1, JobID: "job1", Status: actions_model.StatusSuccess, Needs: []string{}},
 | |
| 				{ID: 2, JobID: "job2", Status: actions_model.StatusBlocked, Needs: []string{"job1"}, WorkflowPayload: []byte(
 | |
| 					`
 | |
| name: test
 | |
| on: push
 | |
| jobs:
 | |
|   job2:
 | |
|     runs-on: ubuntu-latest
 | |
|     needs: job1
 | |
|     if: ${{ always() && needs.job1.result == 'success' }}
 | |
|     steps:
 | |
|       - run: echo "will be checked by act_runner"
 | |
| `)},
 | |
| 			},
 | |
| 			want: map[int64]actions_model.Status{2: actions_model.StatusWaiting},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "`if` is not empty and not all jobs in `needs` completed successfully",
 | |
| 			jobs: actions_model.ActionJobList{
 | |
| 				{ID: 1, JobID: "job1", Status: actions_model.StatusFailure, Needs: []string{}},
 | |
| 				{ID: 2, JobID: "job2", Status: actions_model.StatusBlocked, Needs: []string{"job1"}, WorkflowPayload: []byte(
 | |
| 					`
 | |
| name: test
 | |
| on: push
 | |
| jobs:
 | |
|   job2:
 | |
|     runs-on: ubuntu-latest
 | |
|     needs: job1
 | |
|     if: ${{ always() && needs.job1.result == 'failure' }}
 | |
|     steps:
 | |
|       - run: echo "will be checked by act_runner"
 | |
| `)},
 | |
| 			},
 | |
| 			want: map[int64]actions_model.Status{2: actions_model.StatusWaiting},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "`if` is empty and not all jobs in `needs` completed successfully",
 | |
| 			jobs: actions_model.ActionJobList{
 | |
| 				{ID: 1, JobID: "job1", Status: actions_model.StatusFailure, Needs: []string{}},
 | |
| 				{ID: 2, JobID: "job2", Status: actions_model.StatusBlocked, Needs: []string{"job1"}, WorkflowPayload: []byte(
 | |
| 					`
 | |
| name: test
 | |
| on: push
 | |
| jobs:
 | |
|   job2:
 | |
|     runs-on: ubuntu-latest
 | |
|     needs: job1
 | |
|     steps:
 | |
|       - run: echo "should be skipped"
 | |
| `)},
 | |
| 			},
 | |
| 			want: map[int64]actions_model.Status{2: actions_model.StatusSkipped},
 | |
| 		},
 | |
| 	}
 | |
| 	for _, tt := range tests {
 | |
| 		t.Run(tt.name, func(t *testing.T) {
 | |
| 			r := newJobStatusResolver(tt.jobs, nil)
 | |
| 			assert.Equal(t, tt.want, r.Resolve(t.Context()))
 | |
| 		})
 | |
| 	}
 | |
| }
 |