mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-22 00:14:59 -04:00 
			
		
		
		
	1. use `test.MockVariableValue` as much as possible 2. avoid `time.Sleep` as much as possible
		
			
				
	
	
		
			157 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			157 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // 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 := setting.AppURL + "api/actions"
 | |
| 
 | |
| 	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(t.Context(), 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, ephemeral bool) {
 | |
| 	r.doPing(t)
 | |
| 	resp, err := r.client.runnerServiceClient.Register(t.Context(), connect.NewRequest(&runnerv1.RegisterRequest{
 | |
| 		Name:      name,
 | |
| 		Token:     token,
 | |
| 		Version:   "mock-runner-version",
 | |
| 		Labels:    labels,
 | |
| 		Ephemeral: ephemeral,
 | |
| 	}))
 | |
| 	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, ephemeral bool) {
 | |
| 	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, ephemeral)
 | |
| }
 | |
| 
 | |
| 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(t.Context(), 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
 | |
| }
 | |
| 
 | |
| func (r *mockRunner) execTask(t *testing.T, task *runnerv1.Task, outcome *mockTaskOutcome) {
 | |
| 	for idx, lr := range outcome.logRows {
 | |
| 		resp, err := r.client.runnerServiceClient.UpdateLog(t.Context(), 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(t.Context(), 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)
 | |
| 	}
 | |
| 	resp, err := r.client.runnerServiceClient.UpdateTask(t.Context(), 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)
 | |
| }
 |