Welcome to mirror list, hosted at ThFree Co, Russian Federation.

readiness_test.go « server « service « praefect « internal - gitlab.com/gitlab-org/gitaly.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: b1de057891b39c3768b3ef6122fd248352252ef6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
package server_test

import (
	"context"
	"io"
	"testing"
	"time"

	"github.com/stretchr/testify/assert"
	"github.com/stretchr/testify/require"
	"gitlab.com/gitlab-org/gitaly/v16/internal/gitaly/service/setup"
	"gitlab.com/gitlab-org/gitaly/v16/internal/praefect"
	"gitlab.com/gitlab-org/gitaly/v16/internal/praefect/config"
	"gitlab.com/gitlab-org/gitaly/v16/internal/praefect/service"
	"gitlab.com/gitlab-org/gitaly/v16/internal/testhelper"
	"gitlab.com/gitlab-org/gitaly/v16/internal/testhelper/testcfg"
	"gitlab.com/gitlab-org/gitaly/v16/internal/testhelper/testserver"
	"gitlab.com/gitlab-org/gitaly/v16/proto/go/gitalypb"
	"google.golang.org/protobuf/types/known/durationpb"
)

func TestServer_ReadinessCheck(t *testing.T) {
	t.Parallel()
	stubCheck := func(t *testing.T, triggered chan string, name string) *service.Check {
		return &service.Check{
			Name: name,
			Run: func(ctx context.Context) error {
				_, ok := ctx.Deadline()
				assert.True(t, ok, "the deadline should be set as we provide timeout")
				triggered <- name
				return nil
			},
		}
	}

	const gitalyStorageName = "praefect-internal-0"
	gitalyCfg := testcfg.Build(t, testcfg.WithStorages(gitalyStorageName))
	gitalyAddr := testserver.RunGitalyServer(t, gitalyCfg, nil, setup.RegisterAll, testserver.WithDisablePraefect())

	praefectConf := config.Config{
		SocketPath: testhelper.GetTemporaryGitalySocketFileName(t),
		VirtualStorages: []*config.VirtualStorage{
			{
				Name: "default",
				Nodes: []*config.Node{
					{
						Storage: gitalyStorageName,
						Address: gitalyAddr,
					},
				},
			},
		},
	}
	ctx := testhelper.Context(t)
	triggered := make(chan string, 2)
	grpcPraefectConn, _, cleanup := praefect.RunPraefectServer(t, ctx, praefectConf, praefect.BuildOptions{
		WithChecks: []service.CheckFunc{
			func(conf config.Config, w io.Writer, quiet bool) *service.Check {
				return stubCheck(t, triggered, "1")
			},
			func(conf config.Config, w io.Writer, quiet bool) *service.Check {
				return stubCheck(t, triggered, "2")
			},
		},
	})
	t.Cleanup(cleanup)
	serverClient := gitalypb.NewServerServiceClient(grpcPraefectConn)
	resp, err := serverClient.ReadinessCheck(ctx, &gitalypb.ReadinessCheckRequest{Timeout: durationpb.New(time.Second)})
	require.NoError(t, err)
	assert.NotNil(t, resp.GetOkResponse())
	if !assert.Nil(t, resp.GetFailureResponse()) {
		for _, failure := range resp.GetFailureResponse().GetFailedChecks() {
			assert.Failf(t, "failed check", "%s: %s", failure.Name, failure.ErrorMessage)
		}
	}
	names := make([]string, 0, cap(triggered))
	for i := 0; i < cap(triggered); i++ {
		name := <-triggered
		names = append(names, name)
	}
	require.ElementsMatch(t, []string{"1", "2"}, names, "both tasks should be triggered for an execution")
}

func TestServer_ReadinessCheck_unreachableGitaly(t *testing.T) {
	t.Parallel()

	ctx := testhelper.Context(t)

	// We create a local listener, but immediately close it so that it cannot be reached. This
	// will cause us to allocate a temporary port, which is preferable to using some kind of
	// random address or DNS name as it is guaranteed to resolve and be unreachable.
	listener, addr := testhelper.GetLocalhostListener(t)
	require.NoError(t, listener.Close())

	praefectConf := config.Config{
		SocketPath: testhelper.GetTemporaryGitalySocketFileName(t),
		VirtualStorages: []*config.VirtualStorage{
			{
				Name: "default",
				Nodes: []*config.Node{
					{
						Storage: "praefect-internal-0",
						Address: "tcp://" + addr,
					},
				},
			},
		},
	}

	grpcConn, _, cleanup := praefect.RunPraefectServer(t, ctx, praefectConf, praefect.BuildOptions{})
	t.Cleanup(cleanup)
	client := gitalypb.NewServerServiceClient(grpcConn)

	resp, err := client.ReadinessCheck(ctx, &gitalypb.ReadinessCheckRequest{Timeout: durationpb.New(time.Nanosecond)})
	require.NoError(t, err)
	require.Nil(t, resp.GetOkResponse())
	require.NotNil(t, resp.GetFailureResponse())
	require.Len(t, resp.GetFailureResponse().FailedChecks, 5)
	require.Equal(t, "clock synchronization", resp.GetFailureResponse().FailedChecks[0].Name)
	require.Equal(t, "database read/write", resp.GetFailureResponse().FailedChecks[1].Name)
	require.Equal(t, "gitaly node connectivity & disk access", resp.GetFailureResponse().FailedChecks[2].Name)
	require.Equal(t, "praefect migrations", resp.GetFailureResponse().FailedChecks[3].Name)
	require.Equal(t, "unavailable repositories", resp.GetFailureResponse().FailedChecks[4].Name)
}