mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-25 03:22:36 +00:00 
			
		
		
		
	Improve Actions test (#32883)
This PR adds a mock runner to test more actions features. (cherry picked from commit df98452c0de9d01338f00aa5d85757623523b1fc)
This commit is contained in:
		
					parent
					
						
							
								92ac337263
							
						
					
				
			
			
				commit
				
					
						b18dcd69f2
					
				
			
		
					 3 changed files with 726 additions and 0 deletions
				
			
		
							
								
								
									
										410
									
								
								tests/integration/actions_job_test.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										410
									
								
								tests/integration/actions_job_test.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,410 @@ | ||||||
|  | // Copyright 2024 The Gitea Authors. All rights reserved. | ||||||
|  | // SPDX-License-Identifier: MIT | ||||||
|  | 
 | ||||||
|  | package integration | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"encoding/base64" | ||||||
|  | 	"fmt" | ||||||
|  | 	"net/http" | ||||||
|  | 	"net/url" | ||||||
|  | 	"testing" | ||||||
|  | 	"time" | ||||||
|  | 
 | ||||||
|  | 	actions_model "code.gitea.io/gitea/models/actions" | ||||||
|  | 	auth_model "code.gitea.io/gitea/models/auth" | ||||||
|  | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  | 	user_model "code.gitea.io/gitea/models/user" | ||||||
|  | 	api "code.gitea.io/gitea/modules/structs" | ||||||
|  | 
 | ||||||
|  | 	runnerv1 "code.gitea.io/actions-proto-go/runner/v1" | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func TestJobWithNeeds(t *testing.T) { | ||||||
|  | 	testCases := []struct { | ||||||
|  | 		treePath         string | ||||||
|  | 		fileContent      string | ||||||
|  | 		outcomes         map[string]*mockTaskOutcome | ||||||
|  | 		expectedStatuses map[string]string | ||||||
|  | 	}{ | ||||||
|  | 		{ | ||||||
|  | 			treePath: ".gitea/workflows/job-with-needs.yml", | ||||||
|  | 			fileContent: `name: job-with-needs | ||||||
|  | on:  | ||||||
|  |   push: | ||||||
|  |     paths: | ||||||
|  |       - '.gitea/workflows/job-with-needs.yml' | ||||||
|  | jobs: | ||||||
|  |   job1: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - run: echo job1 | ||||||
|  |   job2: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     needs: [job1] | ||||||
|  |     steps: | ||||||
|  |       - run: echo job2 | ||||||
|  | `, | ||||||
|  | 			outcomes: map[string]*mockTaskOutcome{ | ||||||
|  | 				"job1": { | ||||||
|  | 					result: runnerv1.Result_RESULT_SUCCESS, | ||||||
|  | 				}, | ||||||
|  | 				"job2": { | ||||||
|  | 					result: runnerv1.Result_RESULT_SUCCESS, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 			expectedStatuses: map[string]string{ | ||||||
|  | 				"job1": actions_model.StatusSuccess.String(), | ||||||
|  | 				"job2": actions_model.StatusSuccess.String(), | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			treePath: ".gitea/workflows/job-with-needs-fail.yml", | ||||||
|  | 			fileContent: `name: job-with-needs-fail | ||||||
|  | on:  | ||||||
|  |   push: | ||||||
|  |     paths: | ||||||
|  |       - '.gitea/workflows/job-with-needs-fail.yml' | ||||||
|  | jobs: | ||||||
|  |   job1: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - run: echo job1 | ||||||
|  |   job2: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     needs: [job1] | ||||||
|  |     steps: | ||||||
|  |       - run: echo job2 | ||||||
|  | `, | ||||||
|  | 			outcomes: map[string]*mockTaskOutcome{ | ||||||
|  | 				"job1": { | ||||||
|  | 					result: runnerv1.Result_RESULT_FAILURE, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 			expectedStatuses: map[string]string{ | ||||||
|  | 				"job1": actions_model.StatusFailure.String(), | ||||||
|  | 				"job2": actions_model.StatusSkipped.String(), | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			treePath: ".gitea/workflows/job-with-needs-fail-if.yml", | ||||||
|  | 			fileContent: `name: job-with-needs-fail-if | ||||||
|  | on:  | ||||||
|  |   push: | ||||||
|  |     paths: | ||||||
|  |       - '.gitea/workflows/job-with-needs-fail-if.yml' | ||||||
|  | jobs: | ||||||
|  |   job1: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - run: echo job1 | ||||||
|  |   job2: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     if: ${{ always() }} | ||||||
|  |     needs: [job1] | ||||||
|  |     steps: | ||||||
|  |       - run: echo job2 | ||||||
|  | `, | ||||||
|  | 			outcomes: map[string]*mockTaskOutcome{ | ||||||
|  | 				"job1": { | ||||||
|  | 					result: runnerv1.Result_RESULT_FAILURE, | ||||||
|  | 				}, | ||||||
|  | 				"job2": { | ||||||
|  | 					result: runnerv1.Result_RESULT_SUCCESS, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 			expectedStatuses: map[string]string{ | ||||||
|  | 				"job1": actions_model.StatusFailure.String(), | ||||||
|  | 				"job2": actions_model.StatusSuccess.String(), | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | 	onGiteaRun(t, func(t *testing.T, u *url.URL) { | ||||||
|  | 		user2 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2}) | ||||||
|  | 		session := loginUser(t, user2.Name) | ||||||
|  | 		token := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeWriteRepository, auth_model.AccessTokenScopeWriteUser) | ||||||
|  | 
 | ||||||
|  | 		apiRepo := createActionsTestRepo(t, token, "actions-jobs-with-needs", false) | ||||||
|  | 		runner := newMockRunner() | ||||||
|  | 		runner.registerAsRepoRunner(t, user2.Name, apiRepo.Name, "mock-runner", []string{"ubuntu-latest"}) | ||||||
|  | 
 | ||||||
|  | 		for _, tc := range testCases { | ||||||
|  | 			t.Run(fmt.Sprintf("test %s", tc.treePath), func(t *testing.T) { | ||||||
|  | 				// create the workflow file | ||||||
|  | 				opts := getWorkflowCreateFileOptions(user2, apiRepo.DefaultBranch, fmt.Sprintf("create %s", tc.treePath), tc.fileContent) | ||||||
|  | 				fileResp := createWorkflowFile(t, token, user2.Name, apiRepo.Name, tc.treePath, opts) | ||||||
|  | 
 | ||||||
|  | 				// fetch and execute task | ||||||
|  | 				for i := 0; i < len(tc.outcomes); i++ { | ||||||
|  | 					task := runner.fetchTask(t) | ||||||
|  | 					jobName := getTaskJobNameByTaskID(t, token, user2.Name, apiRepo.Name, task.Id) | ||||||
|  | 					outcome := tc.outcomes[jobName] | ||||||
|  | 					assert.NotNil(t, outcome) | ||||||
|  | 					runner.execTask(t, task, outcome) | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				// check result | ||||||
|  | 				req := NewRequest(t, "GET", fmt.Sprintf("/api/v1/repos/%s/%s/actions/tasks", user2.Name, apiRepo.Name)). | ||||||
|  | 					AddTokenAuth(token) | ||||||
|  | 				resp := MakeRequest(t, req, http.StatusOK) | ||||||
|  | 				var actionTaskRespAfter api.ActionTaskResponse | ||||||
|  | 				DecodeJSON(t, resp, &actionTaskRespAfter) | ||||||
|  | 				for _, apiTask := range actionTaskRespAfter.Entries { | ||||||
|  | 					if apiTask.HeadSHA != fileResp.Commit.SHA { | ||||||
|  | 						continue | ||||||
|  | 					} | ||||||
|  | 					status := apiTask.Status | ||||||
|  | 					assert.Equal(t, status, tc.expectedStatuses[apiTask.Name]) | ||||||
|  | 				} | ||||||
|  | 			}) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		httpContext := NewAPITestContext(t, user2.Name, apiRepo.Name, auth_model.AccessTokenScopeWriteRepository) | ||||||
|  | 		doAPIDeleteRepository(httpContext)(t) | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestJobNeedsMatrix(t *testing.T) { | ||||||
|  | 	testCases := []struct { | ||||||
|  | 		treePath          string | ||||||
|  | 		fileContent       string | ||||||
|  | 		outcomes          map[string]*mockTaskOutcome | ||||||
|  | 		expectedTaskNeeds map[string]*runnerv1.TaskNeed // jobID => TaskNeed | ||||||
|  | 	}{ | ||||||
|  | 		{ | ||||||
|  | 			treePath: ".gitea/workflows/jobs-outputs-with-matrix.yml", | ||||||
|  | 			fileContent: `name: jobs-outputs-with-matrix | ||||||
|  | on:  | ||||||
|  |   push: | ||||||
|  |     paths: | ||||||
|  |       - '.gitea/workflows/jobs-outputs-with-matrix.yml' | ||||||
|  | jobs: | ||||||
|  |   job1: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     outputs: | ||||||
|  |       output_1: ${{ steps.gen_output.outputs.output_1 }} | ||||||
|  |       output_2: ${{ steps.gen_output.outputs.output_2 }} | ||||||
|  |       output_3: ${{ steps.gen_output.outputs.output_3 }} | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         version: [1, 2, 3] | ||||||
|  |     steps: | ||||||
|  |       - name: Generate output | ||||||
|  |         id: gen_output | ||||||
|  |         run: | | ||||||
|  |           version="${{ matrix.version }}" | ||||||
|  |           echo "output_${version}=${version}" >> "$GITHUB_OUTPUT"           | ||||||
|  |   job2: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     needs: [job1] | ||||||
|  |     steps: | ||||||
|  |       - run: echo '${{ toJSON(needs.job1.outputs) }}' | ||||||
|  | `, | ||||||
|  | 			outcomes: map[string]*mockTaskOutcome{ | ||||||
|  | 				"job1 (1)": { | ||||||
|  | 					result: runnerv1.Result_RESULT_SUCCESS, | ||||||
|  | 					outputs: map[string]string{ | ||||||
|  | 						"output_1": "1", | ||||||
|  | 						"output_2": "", | ||||||
|  | 						"output_3": "", | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 				"job1 (2)": { | ||||||
|  | 					result: runnerv1.Result_RESULT_SUCCESS, | ||||||
|  | 					outputs: map[string]string{ | ||||||
|  | 						"output_1": "", | ||||||
|  | 						"output_2": "2", | ||||||
|  | 						"output_3": "", | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 				"job1 (3)": { | ||||||
|  | 					result: runnerv1.Result_RESULT_SUCCESS, | ||||||
|  | 					outputs: map[string]string{ | ||||||
|  | 						"output_1": "", | ||||||
|  | 						"output_2": "", | ||||||
|  | 						"output_3": "3", | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 			expectedTaskNeeds: map[string]*runnerv1.TaskNeed{ | ||||||
|  | 				"job1": { | ||||||
|  | 					Result: runnerv1.Result_RESULT_SUCCESS, | ||||||
|  | 					Outputs: map[string]string{ | ||||||
|  | 						"output_1": "1", | ||||||
|  | 						"output_2": "2", | ||||||
|  | 						"output_3": "3", | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			treePath: ".gitea/workflows/jobs-outputs-with-matrix-failure.yml", | ||||||
|  | 			fileContent: `name: jobs-outputs-with-matrix-failure | ||||||
|  | on:  | ||||||
|  |   push: | ||||||
|  |     paths: | ||||||
|  |       - '.gitea/workflows/jobs-outputs-with-matrix-failure.yml' | ||||||
|  | jobs: | ||||||
|  |   job1: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     outputs: | ||||||
|  |       output_1: ${{ steps.gen_output.outputs.output_1 }} | ||||||
|  |       output_2: ${{ steps.gen_output.outputs.output_2 }} | ||||||
|  |       output_3: ${{ steps.gen_output.outputs.output_3 }} | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         version: [1, 2, 3] | ||||||
|  |     steps: | ||||||
|  |       - name: Generate output | ||||||
|  |         id: gen_output | ||||||
|  |         run: | | ||||||
|  |           version="${{ matrix.version }}" | ||||||
|  |           echo "output_${version}=${version}" >> "$GITHUB_OUTPUT"           | ||||||
|  |   job2: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     if: ${{ always() }} | ||||||
|  |     needs: [job1] | ||||||
|  |     steps: | ||||||
|  |       - run: echo '${{ toJSON(needs.job1.outputs) }}' | ||||||
|  | `, | ||||||
|  | 			outcomes: map[string]*mockTaskOutcome{ | ||||||
|  | 				"job1 (1)": { | ||||||
|  | 					result: runnerv1.Result_RESULT_SUCCESS, | ||||||
|  | 					outputs: map[string]string{ | ||||||
|  | 						"output_1": "1", | ||||||
|  | 						"output_2": "", | ||||||
|  | 						"output_3": "", | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 				"job1 (2)": { | ||||||
|  | 					result: runnerv1.Result_RESULT_FAILURE, | ||||||
|  | 					outputs: map[string]string{ | ||||||
|  | 						"output_1": "", | ||||||
|  | 						"output_2": "", | ||||||
|  | 						"output_3": "", | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 				"job1 (3)": { | ||||||
|  | 					result: runnerv1.Result_RESULT_SUCCESS, | ||||||
|  | 					outputs: map[string]string{ | ||||||
|  | 						"output_1": "", | ||||||
|  | 						"output_2": "", | ||||||
|  | 						"output_3": "3", | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 			expectedTaskNeeds: map[string]*runnerv1.TaskNeed{ | ||||||
|  | 				"job1": { | ||||||
|  | 					Result: runnerv1.Result_RESULT_FAILURE, | ||||||
|  | 					Outputs: map[string]string{ | ||||||
|  | 						"output_1": "1", | ||||||
|  | 						"output_2": "", | ||||||
|  | 						"output_3": "3", | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | 	onGiteaRun(t, func(t *testing.T, u *url.URL) { | ||||||
|  | 		user2 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2}) | ||||||
|  | 		session := loginUser(t, user2.Name) | ||||||
|  | 		token := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeWriteRepository, auth_model.AccessTokenScopeWriteUser) | ||||||
|  | 
 | ||||||
|  | 		apiRepo := createActionsTestRepo(t, token, "actions-jobs-outputs-with-matrix", false) | ||||||
|  | 		runner := newMockRunner() | ||||||
|  | 		runner.registerAsRepoRunner(t, user2.Name, apiRepo.Name, "mock-runner", []string{"ubuntu-latest"}) | ||||||
|  | 
 | ||||||
|  | 		for _, tc := range testCases { | ||||||
|  | 			t.Run(fmt.Sprintf("test %s", tc.treePath), func(t *testing.T) { | ||||||
|  | 				opts := getWorkflowCreateFileOptions(user2, apiRepo.DefaultBranch, fmt.Sprintf("create %s", tc.treePath), tc.fileContent) | ||||||
|  | 				createWorkflowFile(t, token, user2.Name, apiRepo.Name, tc.treePath, opts) | ||||||
|  | 
 | ||||||
|  | 				for i := 0; i < len(tc.outcomes); i++ { | ||||||
|  | 					task := runner.fetchTask(t) | ||||||
|  | 					jobName := getTaskJobNameByTaskID(t, token, user2.Name, apiRepo.Name, task.Id) | ||||||
|  | 					outcome := tc.outcomes[jobName] | ||||||
|  | 					assert.NotNil(t, outcome) | ||||||
|  | 					runner.execTask(t, task, outcome) | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				task := runner.fetchTask(t) | ||||||
|  | 				actualTaskNeeds := task.Needs | ||||||
|  | 				assert.Len(t, actualTaskNeeds, len(tc.expectedTaskNeeds)) | ||||||
|  | 				for jobID, tn := range tc.expectedTaskNeeds { | ||||||
|  | 					actualNeed := actualTaskNeeds[jobID] | ||||||
|  | 					assert.Equal(t, tn.Result, actualNeed.Result) | ||||||
|  | 					assert.Len(t, actualNeed.Outputs, len(tn.Outputs)) | ||||||
|  | 					for outputKey, outputValue := range tn.Outputs { | ||||||
|  | 						assert.Equal(t, outputValue, actualNeed.Outputs[outputKey]) | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			}) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		httpContext := NewAPITestContext(t, user2.Name, apiRepo.Name, auth_model.AccessTokenScopeWriteRepository) | ||||||
|  | 		doAPIDeleteRepository(httpContext)(t) | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func createActionsTestRepo(t *testing.T, authToken, repoName string, isPrivate bool) *api.Repository { | ||||||
|  | 	req := NewRequestWithJSON(t, "POST", "/api/v1/user/repos", &api.CreateRepoOption{ | ||||||
|  | 		Name:          repoName, | ||||||
|  | 		Private:       isPrivate, | ||||||
|  | 		Readme:        "Default", | ||||||
|  | 		AutoInit:      true, | ||||||
|  | 		DefaultBranch: "main", | ||||||
|  | 	}).AddTokenAuth(authToken) | ||||||
|  | 	resp := MakeRequest(t, req, http.StatusCreated) | ||||||
|  | 	var apiRepo api.Repository | ||||||
|  | 	DecodeJSON(t, resp, &apiRepo) | ||||||
|  | 	return &apiRepo | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func getWorkflowCreateFileOptions(u *user_model.User, branch, msg, content string) *api.CreateFileOptions { | ||||||
|  | 	return &api.CreateFileOptions{ | ||||||
|  | 		FileOptions: api.FileOptions{ | ||||||
|  | 			BranchName: branch, | ||||||
|  | 			Message:    msg, | ||||||
|  | 			Author: api.Identity{ | ||||||
|  | 				Name:  u.Name, | ||||||
|  | 				Email: u.Email, | ||||||
|  | 			}, | ||||||
|  | 			Committer: api.Identity{ | ||||||
|  | 				Name:  u.Name, | ||||||
|  | 				Email: u.Email, | ||||||
|  | 			}, | ||||||
|  | 			Dates: api.CommitDateOptions{ | ||||||
|  | 				Author:    time.Now(), | ||||||
|  | 				Committer: time.Now(), | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		ContentBase64: base64.StdEncoding.EncodeToString([]byte(content)), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func createWorkflowFile(t *testing.T, authToken, ownerName, repoName, treePath string, opts *api.CreateFileOptions) *api.FileResponse { | ||||||
|  | 	req := NewRequestWithJSON(t, "POST", fmt.Sprintf("/api/v1/repos/%s/%s/contents/%s", ownerName, repoName, treePath), opts). | ||||||
|  | 		AddTokenAuth(authToken) | ||||||
|  | 	resp := MakeRequest(t, req, http.StatusCreated) | ||||||
|  | 	var fileResponse api.FileResponse | ||||||
|  | 	DecodeJSON(t, resp, &fileResponse) | ||||||
|  | 	return &fileResponse | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // getTaskJobNameByTaskID get the job name of the task by task ID | ||||||
|  | // there is currently not an API for querying a task by ID so we have to list all the tasks | ||||||
|  | func getTaskJobNameByTaskID(t *testing.T, authToken, ownerName, repoName string, taskID int64) string { | ||||||
|  | 	// FIXME: we may need to query several pages | ||||||
|  | 	req := NewRequest(t, "GET", fmt.Sprintf("/api/v1/repos/%s/%s/actions/tasks", ownerName, repoName)). | ||||||
|  | 		AddTokenAuth(authToken) | ||||||
|  | 	resp := MakeRequest(t, req, http.StatusOK) | ||||||
|  | 	var taskRespBefore api.ActionTaskResponse | ||||||
|  | 	DecodeJSON(t, resp, &taskRespBefore) | ||||||
|  | 	for _, apiTask := range taskRespBefore.Entries { | ||||||
|  | 		if apiTask.ID == taskID { | ||||||
|  | 			return apiTask.Name | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return "" | ||||||
|  | } | ||||||
							
								
								
									
										159
									
								
								tests/integration/actions_log_test.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										159
									
								
								tests/integration/actions_log_test.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,159 @@ | ||||||
|  | // Copyright 2024 The Gitea Authors. All rights reserved. | ||||||
|  | // SPDX-License-Identifier: MIT | ||||||
|  | 
 | ||||||
|  | package integration | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"net/http" | ||||||
|  | 	"net/url" | ||||||
|  | 	"strings" | ||||||
|  | 	"testing" | ||||||
|  | 	"time" | ||||||
|  | 
 | ||||||
|  | 	auth_model "code.gitea.io/gitea/models/auth" | ||||||
|  | 	repo_model "code.gitea.io/gitea/models/repo" | ||||||
|  | 	"code.gitea.io/gitea/models/unittest" | ||||||
|  | 	user_model "code.gitea.io/gitea/models/user" | ||||||
|  | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  | 	"code.gitea.io/gitea/modules/storage" | ||||||
|  | 	"code.gitea.io/gitea/modules/test" | ||||||
|  | 
 | ||||||
|  | 	runnerv1 "code.gitea.io/actions-proto-go/runner/v1" | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | 	"google.golang.org/protobuf/types/known/timestamppb" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func TestDownloadTaskLogs(t *testing.T) { | ||||||
|  | 	now := time.Now() | ||||||
|  | 	testCases := []struct { | ||||||
|  | 		treePath    string | ||||||
|  | 		fileContent string | ||||||
|  | 		outcome     *mockTaskOutcome | ||||||
|  | 		zstdEnabled bool | ||||||
|  | 	}{ | ||||||
|  | 		{ | ||||||
|  | 			treePath: ".gitea/workflows/download-task-logs-zstd.yml", | ||||||
|  | 			fileContent: `name: download-task-logs-zstd | ||||||
|  | on:  | ||||||
|  |   push: | ||||||
|  |     paths: | ||||||
|  |       - '.gitea/workflows/download-task-logs-zstd.yml' | ||||||
|  | jobs: | ||||||
|  |     job1: | ||||||
|  |       runs-on: ubuntu-latest | ||||||
|  |       steps: | ||||||
|  |         - run: echo job1 with zstd enabled | ||||||
|  | `, | ||||||
|  | 			outcome: &mockTaskOutcome{ | ||||||
|  | 				result: runnerv1.Result_RESULT_SUCCESS, | ||||||
|  | 				logRows: []*runnerv1.LogRow{ | ||||||
|  | 					{ | ||||||
|  | 						Time:    timestamppb.New(now.Add(1 * time.Second)), | ||||||
|  | 						Content: "  \U0001F433  docker create image", | ||||||
|  | 					}, | ||||||
|  | 					{ | ||||||
|  | 						Time:    timestamppb.New(now.Add(2 * time.Second)), | ||||||
|  | 						Content: "job1 zstd enabled", | ||||||
|  | 					}, | ||||||
|  | 					{ | ||||||
|  | 						Time:    timestamppb.New(now.Add(3 * time.Second)), | ||||||
|  | 						Content: "\U0001F3C1  Job succeeded", | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 			zstdEnabled: true, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			treePath: ".gitea/workflows/download-task-logs-no-zstd.yml", | ||||||
|  | 			fileContent: `name: download-task-logs-no-zstd | ||||||
|  | on:  | ||||||
|  |   push: | ||||||
|  |     paths: | ||||||
|  |       - '.gitea/workflows/download-task-logs-no-zstd.yml' | ||||||
|  | jobs: | ||||||
|  |     job1: | ||||||
|  |       runs-on: ubuntu-latest | ||||||
|  |       steps: | ||||||
|  |         - run: echo job1 with zstd disabled | ||||||
|  | `, | ||||||
|  | 			outcome: &mockTaskOutcome{ | ||||||
|  | 				result: runnerv1.Result_RESULT_SUCCESS, | ||||||
|  | 				logRows: []*runnerv1.LogRow{ | ||||||
|  | 					{ | ||||||
|  | 						Time:    timestamppb.New(now.Add(4 * time.Second)), | ||||||
|  | 						Content: "  \U0001F433  docker create image", | ||||||
|  | 					}, | ||||||
|  | 					{ | ||||||
|  | 						Time:    timestamppb.New(now.Add(5 * time.Second)), | ||||||
|  | 						Content: "job1 zstd disabled", | ||||||
|  | 					}, | ||||||
|  | 					{ | ||||||
|  | 						Time:    timestamppb.New(now.Add(6 * time.Second)), | ||||||
|  | 						Content: "\U0001F3C1  Job succeeded", | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
|  | 			}, | ||||||
|  | 			zstdEnabled: false, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | 	onGiteaRun(t, func(t *testing.T, u *url.URL) { | ||||||
|  | 		user2 := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2}) | ||||||
|  | 		session := loginUser(t, user2.Name) | ||||||
|  | 		token := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeWriteRepository, auth_model.AccessTokenScopeWriteUser) | ||||||
|  | 
 | ||||||
|  | 		apiRepo := createActionsTestRepo(t, token, "actions-download-task-logs", false) | ||||||
|  | 		repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: apiRepo.ID}) | ||||||
|  | 		runner := newMockRunner() | ||||||
|  | 		runner.registerAsRepoRunner(t, user2.Name, repo.Name, "mock-runner", []string{"ubuntu-latest"}) | ||||||
|  | 
 | ||||||
|  | 		for _, tc := range testCases { | ||||||
|  | 			t.Run(fmt.Sprintf("test %s", tc.treePath), func(t *testing.T) { | ||||||
|  | 				var resetFunc func() | ||||||
|  | 				if tc.zstdEnabled { | ||||||
|  | 					resetFunc = test.MockVariableValue(&setting.Actions.LogCompression, "zstd") | ||||||
|  | 					assert.True(t, setting.Actions.LogCompression.IsZstd()) | ||||||
|  | 				} else { | ||||||
|  | 					resetFunc = test.MockVariableValue(&setting.Actions.LogCompression, "none") | ||||||
|  | 					assert.False(t, setting.Actions.LogCompression.IsZstd()) | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				// create the workflow file | ||||||
|  | 				opts := getWorkflowCreateFileOptions(user2, repo.DefaultBranch, fmt.Sprintf("create %s", tc.treePath), tc.fileContent) | ||||||
|  | 				createWorkflowFile(t, token, user2.Name, repo.Name, tc.treePath, opts) | ||||||
|  | 
 | ||||||
|  | 				// fetch and execute task | ||||||
|  | 				task := runner.fetchTask(t) | ||||||
|  | 				runner.execTask(t, task, tc.outcome) | ||||||
|  | 
 | ||||||
|  | 				// check whether the log file exists | ||||||
|  | 				logFileName := fmt.Sprintf("%s/%02x/%d.log", repo.FullName(), task.Id%256, task.Id) | ||||||
|  | 				if setting.Actions.LogCompression.IsZstd() { | ||||||
|  | 					logFileName += ".zst" | ||||||
|  | 				} | ||||||
|  | 				_, err := storage.Actions.Stat(logFileName) | ||||||
|  | 				assert.NoError(t, err) | ||||||
|  | 
 | ||||||
|  | 				// download task logs and check content | ||||||
|  | 				runIndex := task.Context.GetFields()["run_number"].GetStringValue() | ||||||
|  | 				req := NewRequest(t, "GET", fmt.Sprintf("/%s/%s/actions/runs/%s/jobs/0/logs", user2.Name, repo.Name, runIndex)). | ||||||
|  | 					AddTokenAuth(token) | ||||||
|  | 				resp := MakeRequest(t, req, http.StatusOK) | ||||||
|  | 				logTextLines := strings.Split(strings.TrimSpace(resp.Body.String()), "\n") | ||||||
|  | 				assert.Len(t, logTextLines, len(tc.outcome.logRows)) | ||||||
|  | 				for idx, lr := range tc.outcome.logRows { | ||||||
|  | 					assert.Equal( | ||||||
|  | 						t, | ||||||
|  | 						fmt.Sprintf("%s %s", lr.Time.AsTime().Format("2006-01-02T15:04:05.0000000Z07:00"), lr.Content), | ||||||
|  | 						logTextLines[idx], | ||||||
|  | 					) | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				resetFunc() | ||||||
|  | 			}) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		httpContext := NewAPITestContext(t, user2.Name, repo.Name, auth_model.AccessTokenScopeWriteRepository) | ||||||
|  | 		doAPIDeleteRepository(httpContext)(t) | ||||||
|  | 	}) | ||||||
|  | } | ||||||
							
								
								
									
										157
									
								
								tests/integration/actions_runner_test.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										157
									
								
								tests/integration/actions_runner_test.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,157 @@ | ||||||
|  | // Copyright 2024 The Gitea Authors. All rights reserved. | ||||||
|  | // SPDX-License-Identifier: MIT | ||||||
|  | 
 | ||||||
|  | package integration | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  | 	"net/http" | ||||||
|  | 	"testing" | ||||||
|  | 	"time" | ||||||
|  | 
 | ||||||
|  | 	auth_model "code.gitea.io/gitea/models/auth" | ||||||
|  | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  | 
 | ||||||
|  | 	pingv1 "code.gitea.io/actions-proto-go/ping/v1" | ||||||
|  | 	"code.gitea.io/actions-proto-go/ping/v1/pingv1connect" | ||||||
|  | 	runnerv1 "code.gitea.io/actions-proto-go/runner/v1" | ||||||
|  | 	"code.gitea.io/actions-proto-go/runner/v1/runnerv1connect" | ||||||
|  | 	"connectrpc.com/connect" | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | 	"google.golang.org/protobuf/types/known/timestamppb" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type mockRunner struct { | ||||||
|  | 	client *mockRunnerClient | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type mockRunnerClient struct { | ||||||
|  | 	pingServiceClient   pingv1connect.PingServiceClient | ||||||
|  | 	runnerServiceClient runnerv1connect.RunnerServiceClient | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func newMockRunner() *mockRunner { | ||||||
|  | 	client := newMockRunnerClient("", "") | ||||||
|  | 	return &mockRunner{client: client} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func newMockRunnerClient(uuid, token string) *mockRunnerClient { | ||||||
|  | 	baseURL := fmt.Sprintf("%sapi/actions", setting.AppURL) | ||||||
|  | 
 | ||||||
|  | 	opt := connect.WithInterceptors(connect.UnaryInterceptorFunc(func(next connect.UnaryFunc) connect.UnaryFunc { | ||||||
|  | 		return func(ctx context.Context, req connect.AnyRequest) (connect.AnyResponse, error) { | ||||||
|  | 			if uuid != "" { | ||||||
|  | 				req.Header().Set("x-runner-uuid", uuid) | ||||||
|  | 			} | ||||||
|  | 			if token != "" { | ||||||
|  | 				req.Header().Set("x-runner-token", token) | ||||||
|  | 			} | ||||||
|  | 			return next(ctx, req) | ||||||
|  | 		} | ||||||
|  | 	})) | ||||||
|  | 
 | ||||||
|  | 	client := &mockRunnerClient{ | ||||||
|  | 		pingServiceClient:   pingv1connect.NewPingServiceClient(http.DefaultClient, baseURL, opt), | ||||||
|  | 		runnerServiceClient: runnerv1connect.NewRunnerServiceClient(http.DefaultClient, baseURL, opt), | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return client | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (r *mockRunner) doPing(t *testing.T) { | ||||||
|  | 	resp, err := r.client.pingServiceClient.Ping(context.Background(), connect.NewRequest(&pingv1.PingRequest{ | ||||||
|  | 		Data: "mock-runner", | ||||||
|  | 	})) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	assert.Equal(t, "Hello, mock-runner!", resp.Msg.Data) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (r *mockRunner) doRegister(t *testing.T, name, token string, labels []string) { | ||||||
|  | 	r.doPing(t) | ||||||
|  | 	resp, err := r.client.runnerServiceClient.Register(context.Background(), connect.NewRequest(&runnerv1.RegisterRequest{ | ||||||
|  | 		Name:    name, | ||||||
|  | 		Token:   token, | ||||||
|  | 		Version: "mock-runner-version", | ||||||
|  | 		Labels:  labels, | ||||||
|  | 	})) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	r.client = newMockRunnerClient(resp.Msg.Runner.Uuid, resp.Msg.Runner.Token) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (r *mockRunner) registerAsRepoRunner(t *testing.T, ownerName, repoName, runnerName string, labels []string) { | ||||||
|  | 	session := loginUser(t, ownerName) | ||||||
|  | 	token := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeWriteRepository) | ||||||
|  | 	req := NewRequest(t, "GET", fmt.Sprintf("/api/v1/repos/%s/%s/actions/runners/registration-token", ownerName, repoName)).AddTokenAuth(token) | ||||||
|  | 	resp := MakeRequest(t, req, http.StatusOK) | ||||||
|  | 	var registrationToken struct { | ||||||
|  | 		Token string `json:"token"` | ||||||
|  | 	} | ||||||
|  | 	DecodeJSON(t, resp, ®istrationToken) | ||||||
|  | 	r.doRegister(t, runnerName, registrationToken.Token, labels) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (r *mockRunner) fetchTask(t *testing.T, timeout ...time.Duration) *runnerv1.Task { | ||||||
|  | 	fetchTimeout := 10 * time.Second | ||||||
|  | 	if len(timeout) > 0 { | ||||||
|  | 		fetchTimeout = timeout[0] | ||||||
|  | 	} | ||||||
|  | 	ddl := time.Now().Add(fetchTimeout) | ||||||
|  | 	var task *runnerv1.Task | ||||||
|  | 	for time.Now().Before(ddl) { | ||||||
|  | 		resp, err := r.client.runnerServiceClient.FetchTask(context.Background(), connect.NewRequest(&runnerv1.FetchTaskRequest{ | ||||||
|  | 			TasksVersion: 0, | ||||||
|  | 		})) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		if resp.Msg.Task != nil { | ||||||
|  | 			task = resp.Msg.Task | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		time.Sleep(time.Second) | ||||||
|  | 	} | ||||||
|  | 	assert.NotNil(t, task, "failed to fetch a task") | ||||||
|  | 	return task | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type mockTaskOutcome struct { | ||||||
|  | 	result   runnerv1.Result | ||||||
|  | 	outputs  map[string]string | ||||||
|  | 	logRows  []*runnerv1.LogRow | ||||||
|  | 	execTime time.Duration | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (r *mockRunner) execTask(t *testing.T, task *runnerv1.Task, outcome *mockTaskOutcome) { | ||||||
|  | 	for idx, lr := range outcome.logRows { | ||||||
|  | 		resp, err := r.client.runnerServiceClient.UpdateLog(context.Background(), connect.NewRequest(&runnerv1.UpdateLogRequest{ | ||||||
|  | 			TaskId: task.Id, | ||||||
|  | 			Index:  int64(idx), | ||||||
|  | 			Rows:   []*runnerv1.LogRow{lr}, | ||||||
|  | 			NoMore: idx == len(outcome.logRows)-1, | ||||||
|  | 		})) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		assert.EqualValues(t, idx+1, resp.Msg.AckIndex) | ||||||
|  | 	} | ||||||
|  | 	sentOutputKeys := make([]string, 0, len(outcome.outputs)) | ||||||
|  | 	for outputKey, outputValue := range outcome.outputs { | ||||||
|  | 		resp, err := r.client.runnerServiceClient.UpdateTask(context.Background(), connect.NewRequest(&runnerv1.UpdateTaskRequest{ | ||||||
|  | 			State: &runnerv1.TaskState{ | ||||||
|  | 				Id:     task.Id, | ||||||
|  | 				Result: runnerv1.Result_RESULT_UNSPECIFIED, | ||||||
|  | 			}, | ||||||
|  | 			Outputs: map[string]string{outputKey: outputValue}, | ||||||
|  | 		})) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		sentOutputKeys = append(sentOutputKeys, outputKey) | ||||||
|  | 		assert.ElementsMatch(t, sentOutputKeys, resp.Msg.SentOutputs) | ||||||
|  | 	} | ||||||
|  | 	time.Sleep(outcome.execTime) | ||||||
|  | 	resp, err := r.client.runnerServiceClient.UpdateTask(context.Background(), connect.NewRequest(&runnerv1.UpdateTaskRequest{ | ||||||
|  | 		State: &runnerv1.TaskState{ | ||||||
|  | 			Id:        task.Id, | ||||||
|  | 			Result:    outcome.result, | ||||||
|  | 			StoppedAt: timestamppb.Now(), | ||||||
|  | 		}, | ||||||
|  | 	})) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	assert.Equal(t, outcome.result, resp.Msg.State.Result) | ||||||
|  | } | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue