refactor: redis queue backend test cleanup
Summary: - Move existing test under a `testify` Suite as `baseRedisWithServerTestSuite` - Those tests require real redis server. - Add `go.uber.org/mock/mockgen@latest` as dependency - as a tool (Makefile). - in the `go.mod` file. - Mock redis client lives under a `mock` directory under the queue module. - That mock module has an extra hand-written mock in-memory redis-like struct. - Add tests using the mock redis client. - Changed the logic around queue provider creation. - Now the `getNewQueue` returns a Queue provider directly, not an init function to create it. The whole Queue module is close to impossible to test properly because everything is private, everything goes through a struct route. Because of that, we can't test for example what keys are used for given queue. To overcome this, as a first step I removed one step from that hard route by allowing custom calls to create new queue provider. To achieve this, I moved the creation logic into the `getNewQueue` (previously it was `getNewQueueFn`). That changes nothing on that side, everything goes as before, except the `newXXX` call happens directly in that function and not outside that. That made it possible to add extra provider specific parameters to those function (`newXXX`). For example a client on redis. Calling it through the `getNewQueue` function, it gets `nil`. - If the provided client is not `nil`, it will use that instead of the connection string. - If it's `nil` (default behaviour), it creates a new redis client as it did before, no changes to that. The rest of the provider code is unchanged. All these changes were required to make it possible to generate mock clients for providers and use them. For the tests, the existing two test cases are good with redis server, and they need some extra helpers, for example to start a new redis server if required, or waiting on a redis server to be ready to use. These helpers are only required for test cases using real redis server. For better isolation, moved existing test under a testify Suite, and moved them into a new test file called `base_redis_with_server_test.go` because, well they test the code with server. These tests do exactly the same as before, calling the same sub-tests the same way as before, the only change is the structure of the test (remove repetition, scope server related helper functions). Finally, we can create unit tests without redis server. The main focus of this group of tests are higher level overview of operations. With the mock redis client we can set up expectations about used queue names, received values, return value to simulate faulty state. These new unit test functions don't test all functionality, at least it's not aimed for it now. It's more about the possibility of doing that and add extra tests around parts we couldn't test before, for example key. What extra features can test the new unit test group: - What is the received key for given queue? For example using `prefix`, or if all the `SXxx` calls are expected to use `queue_unique` if it's a unique queue. - If it's not a unique queue, no `SXxx` functions are called, because those sets are used only to check if a value is unique or not. - `HasItem` return `false` always if it's a non-unique queue. - All functions are called exactly `N` times, and we don't have any unexpected calls to redis from the code. Signed-off-by: Victoria Nadasdi <victoria@efertone.me>
This commit is contained in:
parent
74e07656d2
commit
54acfa8880
9 changed files with 7581 additions and 117 deletions
3
Makefile
3
Makefile
|
@ -38,6 +38,7 @@ GO_LICENSES_PACKAGE ?= github.com/google/go-licenses@v1.6.0 # renovate: datasour
|
||||||
GOVULNCHECK_PACKAGE ?= golang.org/x/vuln/cmd/govulncheck@v1 # renovate: datasource=go
|
GOVULNCHECK_PACKAGE ?= golang.org/x/vuln/cmd/govulncheck@v1 # renovate: datasource=go
|
||||||
ACTIONLINT_PACKAGE ?= github.com/rhysd/actionlint/cmd/actionlint@v1.6.27 # renovate: datasource=go
|
ACTIONLINT_PACKAGE ?= github.com/rhysd/actionlint/cmd/actionlint@v1.6.27 # renovate: datasource=go
|
||||||
DEADCODE_PACKAGE ?= golang.org/x/tools/internal/cmd/deadcode@v0.14.0 # renovate: datasource=go
|
DEADCODE_PACKAGE ?= golang.org/x/tools/internal/cmd/deadcode@v0.14.0 # renovate: datasource=go
|
||||||
|
GOMOCK_PACKAGE ?= go.uber.org/mock/mockgen@latest
|
||||||
|
|
||||||
DOCKER_IMAGE ?= gitea/gitea
|
DOCKER_IMAGE ?= gitea/gitea
|
||||||
DOCKER_TAG ?= latest
|
DOCKER_TAG ?= latest
|
||||||
|
@ -783,6 +784,7 @@ generate-backend: $(TAGS_PREREQ) generate-go
|
||||||
generate-go: $(TAGS_PREREQ)
|
generate-go: $(TAGS_PREREQ)
|
||||||
@echo "Running go generate..."
|
@echo "Running go generate..."
|
||||||
@CC= GOOS= GOARCH= CGO_ENABLED=0 $(GO) generate -tags '$(TAGS)' ./...
|
@CC= GOOS= GOARCH= CGO_ENABLED=0 $(GO) generate -tags '$(TAGS)' ./...
|
||||||
|
$(GO) run $(GOMOCK_PACKAGE) -package mock -destination ./modules/queue/mock/redisuniversalclient.go github.com/redis/go-redis/v9 UniversalClient
|
||||||
|
|
||||||
.PHONY: merge-locales
|
.PHONY: merge-locales
|
||||||
merge-locales:
|
merge-locales:
|
||||||
|
@ -884,6 +886,7 @@ deps-tools:
|
||||||
$(GO) install $(GO_LICENSES_PACKAGE)
|
$(GO) install $(GO_LICENSES_PACKAGE)
|
||||||
$(GO) install $(GOVULNCHECK_PACKAGE)
|
$(GO) install $(GOVULNCHECK_PACKAGE)
|
||||||
$(GO) install $(ACTIONLINT_PACKAGE)
|
$(GO) install $(ACTIONLINT_PACKAGE)
|
||||||
|
$(GO) install $(GOMOCK_PACKAGE)
|
||||||
|
|
||||||
node_modules: package-lock.json
|
node_modules: package-lock.json
|
||||||
npm install --no-save
|
npm install --no-save
|
||||||
|
|
1
go.mod
1
go.mod
|
@ -281,6 +281,7 @@ require (
|
||||||
go.opentelemetry.io/otel v1.22.0 // indirect
|
go.opentelemetry.io/otel v1.22.0 // indirect
|
||||||
go.opentelemetry.io/otel/trace v1.22.0 // indirect
|
go.opentelemetry.io/otel/trace v1.22.0 // indirect
|
||||||
go.uber.org/atomic v1.11.0 // indirect
|
go.uber.org/atomic v1.11.0 // indirect
|
||||||
|
go.uber.org/mock v0.4.0 // indirect
|
||||||
go.uber.org/multierr v1.11.0 // indirect
|
go.uber.org/multierr v1.11.0 // indirect
|
||||||
go.uber.org/zap v1.27.0 // indirect
|
go.uber.org/zap v1.27.0 // indirect
|
||||||
golang.org/x/exp v0.0.0-20240119083558-1b970713d09a // indirect
|
golang.org/x/exp v0.0.0-20240119083558-1b970713d09a // indirect
|
||||||
|
|
2
go.sum
2
go.sum
|
@ -877,6 +877,8 @@ go.uber.org/atomic v1.11.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0
|
||||||
go.uber.org/goleak v1.1.10/go.mod h1:8a7PlsEVH3e/a/GLqe5IIrQx6GzcnRmZEufDUTk4A7A=
|
go.uber.org/goleak v1.1.10/go.mod h1:8a7PlsEVH3e/a/GLqe5IIrQx6GzcnRmZEufDUTk4A7A=
|
||||||
go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto=
|
go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto=
|
||||||
go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE=
|
go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE=
|
||||||
|
go.uber.org/mock v0.4.0 h1:VcM4ZOtdbR4f6VXfiOpwpVJDL6lCReaZ6mw31wqh7KU=
|
||||||
|
go.uber.org/mock v0.4.0/go.mod h1:a6FSlNadKUHUa9IP5Vyt1zh4fC7uAwxMutEAscFbkZc=
|
||||||
go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
|
go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
|
||||||
go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
|
go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
|
||||||
go.uber.org/zap v1.27.0 h1:aJMhYGrd5QSmlpLMr2MftRKl7t8J8PTZPA732ud/XR8=
|
go.uber.org/zap v1.27.0 h1:aJMhYGrd5QSmlpLMr2MftRKl7t8J8PTZPA732ud/XR8=
|
||||||
|
|
|
@ -26,8 +26,11 @@ type baseRedis struct {
|
||||||
|
|
||||||
var _ baseQueue = (*baseRedis)(nil)
|
var _ baseQueue = (*baseRedis)(nil)
|
||||||
|
|
||||||
func newBaseRedisGeneric(cfg *BaseConfig, unique bool) (baseQueue, error) {
|
func newBaseRedisGeneric(cfg *BaseConfig, unique bool, client redis.UniversalClient) (baseQueue, error) {
|
||||||
client := nosql.GetManager().GetRedisClient(cfg.ConnStr)
|
if client == nil {
|
||||||
|
client = nosql.GetManager().GetRedisClient(cfg.ConnStr)
|
||||||
|
}
|
||||||
|
|
||||||
prefix := ""
|
prefix := ""
|
||||||
uri := nosql.ToRedisURI(cfg.ConnStr)
|
uri := nosql.ToRedisURI(cfg.ConnStr)
|
||||||
|
|
||||||
|
@ -62,11 +65,11 @@ func newBaseRedisGeneric(cfg *BaseConfig, unique bool) (baseQueue, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func newBaseRedisSimple(cfg *BaseConfig) (baseQueue, error) {
|
func newBaseRedisSimple(cfg *BaseConfig) (baseQueue, error) {
|
||||||
return newBaseRedisGeneric(cfg, false)
|
return newBaseRedisGeneric(cfg, false, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
func newBaseRedisUnique(cfg *BaseConfig) (baseQueue, error) {
|
func newBaseRedisUnique(cfg *BaseConfig) (baseQueue, error) {
|
||||||
return newBaseRedisGeneric(cfg, true)
|
return newBaseRedisGeneric(cfg, true, nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (q *baseRedis) prefixedName(name string) string {
|
func (q *baseRedis) prefixedName(name string) string {
|
||||||
|
|
|
@ -5,120 +5,134 @@ package queue
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"os"
|
|
||||||
"os/exec"
|
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/nosql"
|
"code.gitea.io/gitea/modules/queue/mock"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/redis/go-redis/v9"
|
||||||
|
"github.com/stretchr/testify/suite"
|
||||||
|
"go.uber.org/mock/gomock"
|
||||||
)
|
)
|
||||||
|
|
||||||
const defaultTestRedisServer = "127.0.0.1:6379"
|
type baseRedisUnitTestSuite struct {
|
||||||
|
suite.Suite
|
||||||
|
|
||||||
func testRedisHost() string {
|
mockController *gomock.Controller
|
||||||
value := os.Getenv("TEST_REDIS_SERVER")
|
|
||||||
if value != "" {
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
|
|
||||||
return defaultTestRedisServer
|
|
||||||
}
|
|
||||||
|
|
||||||
func waitRedisReady(conn string, dur time.Duration) (ready bool) {
|
|
||||||
ctxTimed, cancel := context.WithTimeout(context.Background(), time.Second*5)
|
|
||||||
defer cancel()
|
|
||||||
for t := time.Now(); ; time.Sleep(50 * time.Millisecond) {
|
|
||||||
ret := nosql.GetManager().GetRedisClient(conn).Ping(ctxTimed)
|
|
||||||
if ret.Err() == nil {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
if time.Since(t) > dur {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func redisServerCmd(t *testing.T) *exec.Cmd {
|
|
||||||
redisServerProg, err := exec.LookPath("redis-server")
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
c := &exec.Cmd{
|
|
||||||
Path: redisServerProg,
|
|
||||||
Args: []string{redisServerProg, "--bind", "127.0.0.1", "--port", "6379"},
|
|
||||||
Dir: t.TempDir(),
|
|
||||||
Stdin: os.Stdin,
|
|
||||||
Stdout: os.Stdout,
|
|
||||||
Stderr: os.Stderr,
|
|
||||||
}
|
|
||||||
return c
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestBaseRedis(t *testing.T) {
|
func TestBaseRedis(t *testing.T) {
|
||||||
redisAddress := "redis://" + testRedisHost() + "/0"
|
suite.Run(t, &baseRedisUnitTestSuite{})
|
||||||
queueSettings := setting.QueueSettings{
|
|
||||||
Length: 10,
|
|
||||||
ConnStr: redisAddress,
|
|
||||||
}
|
|
||||||
|
|
||||||
var redisServer *exec.Cmd
|
|
||||||
if !waitRedisReady(redisAddress, 0) {
|
|
||||||
redisServer = redisServerCmd(t)
|
|
||||||
|
|
||||||
if redisServer == nil {
|
|
||||||
t.Skip("redis-server not found in Forgejo test yet")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
assert.NoError(t, redisServer.Start())
|
|
||||||
if !assert.True(t, waitRedisReady(redisAddress, 5*time.Second), "start redis-server") {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
defer func() {
|
|
||||||
if redisServer != nil {
|
|
||||||
_ = redisServer.Process.Signal(os.Interrupt)
|
|
||||||
_ = redisServer.Wait()
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
testQueueBasic(t, newBaseRedisSimple, toBaseConfig("baseRedis", queueSettings), false)
|
|
||||||
testQueueBasic(t, newBaseRedisUnique, toBaseConfig("baseRedisUnique", queueSettings), true)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestBaseRedisWithPrefix(t *testing.T) {
|
func (suite *baseRedisUnitTestSuite) SetupSuite() {
|
||||||
redisAddress := "redis://" + testRedisHost() + "/0?prefix=forgejo:queue:"
|
suite.mockController = gomock.NewController(suite.T())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *baseRedisUnitTestSuite) TestBasic() {
|
||||||
|
queueName := "test-queue"
|
||||||
|
testCases := []struct {
|
||||||
|
Name string
|
||||||
|
ConnectionString string
|
||||||
|
QueueName string
|
||||||
|
Unique bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
Name: "unique",
|
||||||
|
ConnectionString: "redis://127.0.0.1/0",
|
||||||
|
QueueName: queueName,
|
||||||
|
Unique: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "non-unique",
|
||||||
|
ConnectionString: "redis://127.0.0.1/0",
|
||||||
|
QueueName: queueName,
|
||||||
|
Unique: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "unique with prefix",
|
||||||
|
ConnectionString: "redis://127.0.0.1/0?prefix=forgejo:queue:",
|
||||||
|
QueueName: "forgejo:queue:" + queueName,
|
||||||
|
Unique: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "non-unique with prefix",
|
||||||
|
ConnectionString: "redis://127.0.0.1/0?prefix=forgejo:queue:",
|
||||||
|
QueueName: "forgejo:queue:" + queueName,
|
||||||
|
Unique: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, testCase := range testCases {
|
||||||
|
suite.Run(testCase.Name, func() {
|
||||||
queueSettings := setting.QueueSettings{
|
queueSettings := setting.QueueSettings{
|
||||||
Length: 10,
|
Length: 10,
|
||||||
ConnStr: redisAddress,
|
ConnStr: testCase.ConnectionString,
|
||||||
}
|
}
|
||||||
|
|
||||||
var redisServer *exec.Cmd
|
// Configure expectations.
|
||||||
if !waitRedisReady(redisAddress, 0) {
|
mockRedisStore := mock.NewInMemoryMockRedis()
|
||||||
redisServer = redisServerCmd(t)
|
redisClient := mock.NewMockUniversalClient(suite.mockController)
|
||||||
|
|
||||||
if redisServer == nil {
|
redisClient.EXPECT().
|
||||||
t.Skip("redis-server not found in Forgejo test yet")
|
Ping(gomock.Any()).
|
||||||
return
|
Times(1).
|
||||||
|
Return(&redis.StatusCmd{})
|
||||||
|
redisClient.EXPECT().
|
||||||
|
LLen(gomock.Any(), testCase.QueueName).
|
||||||
|
Times(1).
|
||||||
|
DoAndReturn(mockRedisStore.LLen)
|
||||||
|
redisClient.EXPECT().
|
||||||
|
LPop(gomock.Any(), testCase.QueueName).
|
||||||
|
Times(1).
|
||||||
|
DoAndReturn(mockRedisStore.LPop)
|
||||||
|
redisClient.EXPECT().
|
||||||
|
RPush(gomock.Any(), testCase.QueueName, gomock.Any()).
|
||||||
|
Times(1).
|
||||||
|
DoAndReturn(mockRedisStore.RPush)
|
||||||
|
|
||||||
|
if testCase.Unique {
|
||||||
|
redisClient.EXPECT().
|
||||||
|
SAdd(gomock.Any(), testCase.QueueName+"_unique", gomock.Any()).
|
||||||
|
Times(1).
|
||||||
|
DoAndReturn(mockRedisStore.SAdd)
|
||||||
|
redisClient.EXPECT().
|
||||||
|
SRem(gomock.Any(), testCase.QueueName+"_unique", gomock.Any()).
|
||||||
|
Times(1).
|
||||||
|
DoAndReturn(mockRedisStore.SRem)
|
||||||
|
redisClient.EXPECT().
|
||||||
|
SIsMember(gomock.Any(), testCase.QueueName+"_unique", gomock.Any()).
|
||||||
|
Times(2).
|
||||||
|
DoAndReturn(mockRedisStore.SIsMember)
|
||||||
}
|
}
|
||||||
|
|
||||||
assert.NoError(t, redisServer.Start())
|
client, err := newBaseRedisGeneric(
|
||||||
if !assert.True(t, waitRedisReady(redisAddress, 5*time.Second), "start redis-server") {
|
toBaseConfig(queueName, queueSettings),
|
||||||
return
|
testCase.Unique,
|
||||||
}
|
redisClient,
|
||||||
|
)
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
expectedContent := []byte("test")
|
||||||
|
|
||||||
|
suite.Require().NoError(client.PushItem(ctx, expectedContent))
|
||||||
|
|
||||||
|
found, err := client.HasItem(ctx, expectedContent)
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
if testCase.Unique {
|
||||||
|
suite.True(found)
|
||||||
|
} else {
|
||||||
|
suite.False(found)
|
||||||
}
|
}
|
||||||
|
|
||||||
defer func() {
|
found, err = client.HasItem(ctx, []byte("not found content"))
|
||||||
if redisServer != nil {
|
suite.Require().NoError(err)
|
||||||
_ = redisServer.Process.Signal(os.Interrupt)
|
suite.False(found)
|
||||||
_ = redisServer.Wait()
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
testQueueBasic(t, newBaseRedisSimple, toBaseConfig("baseRedis", queueSettings), false)
|
content, err := client.PopItem(ctx)
|
||||||
testQueueBasic(t, newBaseRedisUnique, toBaseConfig("baseRedisUnique", queueSettings), true)
|
suite.Require().NoError(err)
|
||||||
|
suite.Equal(expectedContent, content)
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
132
modules/queue/base_redis_with_server_test.go
Normal file
132
modules/queue/base_redis_with_server_test.go
Normal file
|
@ -0,0 +1,132 @@
|
||||||
|
package queue
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/modules/nosql"
|
||||||
|
"code.gitea.io/gitea/modules/setting"
|
||||||
|
"github.com/stretchr/testify/suite"
|
||||||
|
)
|
||||||
|
|
||||||
|
const defaultTestRedisServer = "127.0.0.1:6379"
|
||||||
|
|
||||||
|
type baseRedisWithServerTestSuite struct {
|
||||||
|
suite.Suite
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestBaseRedisWithServer(t *testing.T) {
|
||||||
|
suite.Run(t, &baseRedisWithServerTestSuite{})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *baseRedisWithServerTestSuite) TestNormal() {
|
||||||
|
redisAddress := "redis://" + suite.testRedisHost() + "/0"
|
||||||
|
queueSettings := setting.QueueSettings{
|
||||||
|
Length: 10,
|
||||||
|
ConnStr: redisAddress,
|
||||||
|
}
|
||||||
|
|
||||||
|
redisServer, accessible := suite.startRedisServer(redisAddress)
|
||||||
|
|
||||||
|
// If it's accessible, but redisServer command is nil, that means we are using
|
||||||
|
// an already running redis server.
|
||||||
|
if redisServer == nil && !accessible {
|
||||||
|
suite.T().Skip("redis-server not found in Forgejo test yet")
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
if redisServer != nil {
|
||||||
|
_ = redisServer.Process.Signal(os.Interrupt)
|
||||||
|
_ = redisServer.Wait()
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
testQueueBasic(suite.T(), newBaseRedisSimple, toBaseConfig("baseRedis", queueSettings), false)
|
||||||
|
testQueueBasic(suite.T(), newBaseRedisUnique, toBaseConfig("baseRedisUnique", queueSettings), true)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *baseRedisWithServerTestSuite) TestWithPrefix() {
|
||||||
|
redisAddress := "redis://" + suite.testRedisHost() + "/0?prefix=forgejo:queue:"
|
||||||
|
queueSettings := setting.QueueSettings{
|
||||||
|
Length: 10,
|
||||||
|
ConnStr: redisAddress,
|
||||||
|
}
|
||||||
|
|
||||||
|
redisServer, accessible := suite.startRedisServer(redisAddress)
|
||||||
|
|
||||||
|
// If it's accessible, but redisServer command is nil, that means we are using
|
||||||
|
// an already running redis server.
|
||||||
|
if redisServer == nil && !accessible {
|
||||||
|
suite.T().Skip("redis-server not found in Forgejo test yet")
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
if redisServer != nil {
|
||||||
|
_ = redisServer.Process.Signal(os.Interrupt)
|
||||||
|
_ = redisServer.Wait()
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
testQueueBasic(suite.T(), newBaseRedisSimple, toBaseConfig("baseRedis", queueSettings), false)
|
||||||
|
testQueueBasic(suite.T(), newBaseRedisUnique, toBaseConfig("baseRedisUnique", queueSettings), true)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *baseRedisWithServerTestSuite) startRedisServer(address string) (*exec.Cmd, bool) {
|
||||||
|
var redisServer *exec.Cmd
|
||||||
|
|
||||||
|
if !suite.waitRedisReady(address, 0) {
|
||||||
|
redisServerProg, err := exec.LookPath("redis-server")
|
||||||
|
if err != nil {
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
redisServer = &exec.Cmd{
|
||||||
|
Path: redisServerProg,
|
||||||
|
Args: []string{redisServerProg, "--bind", "127.0.0.1", "--port", "6379"},
|
||||||
|
Dir: suite.T().TempDir(),
|
||||||
|
Stdin: os.Stdin,
|
||||||
|
Stdout: os.Stdout,
|
||||||
|
Stderr: os.Stderr,
|
||||||
|
}
|
||||||
|
|
||||||
|
suite.Require().NoError(redisServer.Start())
|
||||||
|
|
||||||
|
if !suite.True(suite.waitRedisReady(address, 5*time.Second), "start redis-server") {
|
||||||
|
// Return with redis server even if it's not available. It was started,
|
||||||
|
// even if it's not reachable for any reasons, it's still started, the
|
||||||
|
// parent will close it.
|
||||||
|
return redisServer, false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return redisServer, true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *baseRedisWithServerTestSuite) waitRedisReady(conn string, dur time.Duration) (ready bool) {
|
||||||
|
ctxTimed, cancel := context.WithTimeout(context.Background(), time.Second*5)
|
||||||
|
defer cancel()
|
||||||
|
for t := time.Now(); ; time.Sleep(50 * time.Millisecond) {
|
||||||
|
ret := nosql.GetManager().GetRedisClient(conn).Ping(ctxTimed)
|
||||||
|
if ret.Err() == nil {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if time.Since(t) > dur {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *baseRedisWithServerTestSuite) testRedisHost() string {
|
||||||
|
value := os.Getenv("TEST_REDIS_SERVER")
|
||||||
|
if value != "" {
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
|
||||||
|
return defaultTestRedisServer
|
||||||
|
}
|
133
modules/queue/mock/inmemorymockredis.go
Normal file
133
modules/queue/mock/inmemorymockredis.go
Normal file
|
@ -0,0 +1,133 @@
|
||||||
|
package mock
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
redis "github.com/redis/go-redis/v9"
|
||||||
|
)
|
||||||
|
|
||||||
|
// InMemoryMockRedis is a very primitive in-memory redis-like feature. The main
|
||||||
|
// purpose of this struct is to give some backend to mocked unit tests.
|
||||||
|
type InMemoryMockRedis struct {
|
||||||
|
queues map[string][][]byte
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewInMemoryMockRedis() InMemoryMockRedis {
|
||||||
|
return InMemoryMockRedis{
|
||||||
|
queues: map[string][][]byte{},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *InMemoryMockRedis) LLen(ctx context.Context, key string) *redis.IntCmd {
|
||||||
|
cmd := redis.NewIntCmd(ctx)
|
||||||
|
cmd.SetVal(int64(len(r.queues[key])))
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *InMemoryMockRedis) SAdd(ctx context.Context, key string, content []byte) *redis.IntCmd {
|
||||||
|
cmd := redis.NewIntCmd(ctx)
|
||||||
|
|
||||||
|
for _, value := range r.queues[key] {
|
||||||
|
if string(value) == string(content) {
|
||||||
|
cmd.SetVal(0)
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
r.queues[key] = append(r.queues[key], content)
|
||||||
|
|
||||||
|
cmd.SetVal(1)
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *InMemoryMockRedis) RPush(ctx context.Context, key string, content []byte) *redis.IntCmd {
|
||||||
|
cmd := redis.NewIntCmd(ctx)
|
||||||
|
|
||||||
|
r.queues[key] = append(r.queues[key], content)
|
||||||
|
|
||||||
|
cmd.SetVal(1)
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *InMemoryMockRedis) LPop(ctx context.Context, key string) *redis.StringCmd {
|
||||||
|
cmd := redis.NewStringCmd(ctx)
|
||||||
|
|
||||||
|
queue, found := r.queues[key]
|
||||||
|
if !found {
|
||||||
|
cmd.SetErr(errors.New("queue not found"))
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(queue) < 1 {
|
||||||
|
cmd.SetErr(errors.New("queue is empty"))
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
value, rest := queue[0], queue[1:]
|
||||||
|
|
||||||
|
r.queues[key] = rest
|
||||||
|
|
||||||
|
cmd.SetVal(string(value))
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *InMemoryMockRedis) SRem(ctx context.Context, key string, content []byte) *redis.IntCmd {
|
||||||
|
cmd := redis.NewIntCmd(ctx)
|
||||||
|
|
||||||
|
queue, found := r.queues[key]
|
||||||
|
if !found {
|
||||||
|
cmd.SetErr(errors.New("queue not found"))
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(queue) < 1 {
|
||||||
|
cmd.SetErr(errors.New("queue is empty"))
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
newList := [][]byte{}
|
||||||
|
|
||||||
|
for _, value := range queue {
|
||||||
|
if string(value) != string(content) {
|
||||||
|
newList = append(newList, value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
r.queues[key] = newList
|
||||||
|
|
||||||
|
cmd.SetVal(1)
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *InMemoryMockRedis) SIsMember(ctx context.Context, key string, content []byte) *redis.BoolCmd {
|
||||||
|
cmd := redis.NewBoolCmd(ctx)
|
||||||
|
|
||||||
|
queue, found := r.queues[key]
|
||||||
|
if !found {
|
||||||
|
cmd.SetErr(errors.New("queue not found"))
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, value := range queue {
|
||||||
|
if string(value) == string(content) {
|
||||||
|
cmd.SetVal(true)
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd.SetVal(false)
|
||||||
|
|
||||||
|
return cmd
|
||||||
|
}
|
7168
modules/queue/mock/redisuniversalclient.go
Normal file
7168
modules/queue/mock/redisuniversalclient.go
Normal file
File diff suppressed because it is too large
Load diff
|
@ -192,16 +192,24 @@ func (q *WorkerPoolQueue[T]) ShutdownWait(timeout time.Duration) {
|
||||||
<-q.shutdownDone
|
<-q.shutdownDone
|
||||||
}
|
}
|
||||||
|
|
||||||
func getNewQueueFn(t string) (string, func(cfg *BaseConfig, unique bool) (baseQueue, error)) {
|
func getNewQueue(t string, cfg *BaseConfig, unique bool) (string, baseQueue, error) {
|
||||||
switch t {
|
switch t {
|
||||||
case "dummy", "immediate":
|
case "dummy", "immediate":
|
||||||
return t, newBaseDummy
|
queue, err := newBaseDummy(cfg, unique)
|
||||||
|
|
||||||
|
return t, queue, err
|
||||||
case "channel":
|
case "channel":
|
||||||
return t, newBaseChannelGeneric
|
queue, err := newBaseChannelGeneric(cfg, unique)
|
||||||
|
|
||||||
|
return t, queue, err
|
||||||
case "redis":
|
case "redis":
|
||||||
return t, newBaseRedisGeneric
|
queue, err := newBaseRedisGeneric(cfg, unique, nil)
|
||||||
|
|
||||||
|
return t, queue, err
|
||||||
default: // level(leveldb,levelqueue,persistable-channel)
|
default: // level(leveldb,levelqueue,persistable-channel)
|
||||||
return "level", newBaseLevelQueueGeneric
|
queue, err := newBaseLevelQueueGeneric(cfg, unique)
|
||||||
|
|
||||||
|
return "level", queue, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -217,14 +225,14 @@ func NewWorkerPoolQueueWithContext[T any](ctx context.Context, name string, queu
|
||||||
|
|
||||||
var w WorkerPoolQueue[T]
|
var w WorkerPoolQueue[T]
|
||||||
var err error
|
var err error
|
||||||
queueType, newQueueFn := getNewQueueFn(queueSetting.Type)
|
|
||||||
w.baseQueueType = queueType
|
|
||||||
w.baseConfig = toBaseConfig(name, queueSetting)
|
w.baseConfig = toBaseConfig(name, queueSetting)
|
||||||
w.baseQueue, err = newQueueFn(w.baseConfig, unique)
|
|
||||||
|
w.baseQueueType, w.baseQueue, err = getNewQueue(queueSetting.Type, w.baseConfig, unique)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
log.Trace("Created queue %q of type %q", name, queueType)
|
log.Trace("Created queue %q of type %q", name, w.baseQueueType)
|
||||||
|
|
||||||
w.ctxRun, _, w.ctxRunCancel = process.GetManager().AddTypedContext(ctx, "Queue: "+w.GetName(), process.SystemProcessType, false)
|
w.ctxRun, _, w.ctxRunCancel = process.GetManager().AddTypedContext(ctx, "Queue: "+w.GetName(), process.SystemProcessType, false)
|
||||||
w.batchChan = make(chan []T)
|
w.batchChan = make(chan []T)
|
||||||
|
|
Loading…
Reference in a new issue