mirror of
				https://github.com/zitadel/zitadel.git
				synced 2025-10-25 20:38:48 +00:00 
			
		
		
		
	feat(cache): redis cache (#8822)
# Which Problems Are Solved Add a cache implementation using Redis single mode. This does not add support for Redis Cluster or sentinel. # How the Problems Are Solved Added the `internal/cache/redis` package. All operations occur atomically, including setting of secondary indexes, using LUA scripts where needed. The [`miniredis`](https://github.com/alicebob/miniredis) package is used to run unit tests. # Additional Changes - Move connector code to `internal/cache/connector/...` and remove duplicate code from `query` and `command` packages. - Fix a missed invalidation on the restrictions projection # Additional Context Closes #8130
This commit is contained in:
		
							
								
								
									
										329
									
								
								internal/cache/connector/gomap/gomap_test.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										329
									
								
								internal/cache/connector/gomap/gomap_test.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,329 @@ | ||||
| package gomap | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| 	"github.com/stretchr/testify/require" | ||||
| 	"github.com/zitadel/logging" | ||||
|  | ||||
| 	"github.com/zitadel/zitadel/internal/cache" | ||||
| ) | ||||
|  | ||||
| type testIndex int | ||||
|  | ||||
| const ( | ||||
| 	testIndexID testIndex = iota | ||||
| 	testIndexName | ||||
| ) | ||||
|  | ||||
| var testIndices = []testIndex{ | ||||
| 	testIndexID, | ||||
| 	testIndexName, | ||||
| } | ||||
|  | ||||
| type testObject struct { | ||||
| 	id    string | ||||
| 	names []string | ||||
| } | ||||
|  | ||||
| func (o *testObject) Keys(index testIndex) []string { | ||||
| 	switch index { | ||||
| 	case testIndexID: | ||||
| 		return []string{o.id} | ||||
| 	case testIndexName: | ||||
| 		return o.names | ||||
| 	default: | ||||
| 		return nil | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func Test_mapCache_Get(t *testing.T) { | ||||
| 	c := NewCache[testIndex, string, *testObject](context.Background(), testIndices, cache.Config{ | ||||
| 		MaxAge:     time.Second, | ||||
| 		LastUseAge: time.Second / 4, | ||||
| 		Log: &logging.Config{ | ||||
| 			Level:     "debug", | ||||
| 			AddSource: true, | ||||
| 		}, | ||||
| 	}) | ||||
| 	obj := &testObject{ | ||||
| 		id:    "id", | ||||
| 		names: []string{"foo", "bar"}, | ||||
| 	} | ||||
| 	c.Set(context.Background(), obj) | ||||
|  | ||||
| 	type args struct { | ||||
| 		index testIndex | ||||
| 		key   string | ||||
| 	} | ||||
| 	tests := []struct { | ||||
| 		name   string | ||||
| 		args   args | ||||
| 		want   *testObject | ||||
| 		wantOk bool | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name: "ok", | ||||
| 			args: args{ | ||||
| 				index: testIndexID, | ||||
| 				key:   "id", | ||||
| 			}, | ||||
| 			want:   obj, | ||||
| 			wantOk: true, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "miss", | ||||
| 			args: args{ | ||||
| 				index: testIndexID, | ||||
| 				key:   "spanac", | ||||
| 			}, | ||||
| 			want:   nil, | ||||
| 			wantOk: false, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "unknown index", | ||||
| 			args: args{ | ||||
| 				index: 99, | ||||
| 				key:   "id", | ||||
| 			}, | ||||
| 			want:   nil, | ||||
| 			wantOk: false, | ||||
| 		}, | ||||
| 	} | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			got, ok := c.Get(context.Background(), tt.args.index, tt.args.key) | ||||
| 			assert.Equal(t, tt.want, got) | ||||
| 			assert.Equal(t, tt.wantOk, ok) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func Test_mapCache_Invalidate(t *testing.T) { | ||||
| 	c := NewCache[testIndex, string, *testObject](context.Background(), testIndices, cache.Config{ | ||||
| 		MaxAge:     time.Second, | ||||
| 		LastUseAge: time.Second / 4, | ||||
| 		Log: &logging.Config{ | ||||
| 			Level:     "debug", | ||||
| 			AddSource: true, | ||||
| 		}, | ||||
| 	}) | ||||
| 	obj := &testObject{ | ||||
| 		id:    "id", | ||||
| 		names: []string{"foo", "bar"}, | ||||
| 	} | ||||
| 	c.Set(context.Background(), obj) | ||||
| 	err := c.Invalidate(context.Background(), testIndexName, "bar") | ||||
| 	require.NoError(t, err) | ||||
| 	got, ok := c.Get(context.Background(), testIndexID, "id") | ||||
| 	assert.Nil(t, got) | ||||
| 	assert.False(t, ok) | ||||
| } | ||||
|  | ||||
| func Test_mapCache_Delete(t *testing.T) { | ||||
| 	c := NewCache[testIndex, string, *testObject](context.Background(), testIndices, cache.Config{ | ||||
| 		MaxAge:     time.Second, | ||||
| 		LastUseAge: time.Second / 4, | ||||
| 		Log: &logging.Config{ | ||||
| 			Level:     "debug", | ||||
| 			AddSource: true, | ||||
| 		}, | ||||
| 	}) | ||||
| 	obj := &testObject{ | ||||
| 		id:    "id", | ||||
| 		names: []string{"foo", "bar"}, | ||||
| 	} | ||||
| 	c.Set(context.Background(), obj) | ||||
| 	err := c.Delete(context.Background(), testIndexName, "bar") | ||||
| 	require.NoError(t, err) | ||||
|  | ||||
| 	// Shouldn't find object by deleted name | ||||
| 	got, ok := c.Get(context.Background(), testIndexName, "bar") | ||||
| 	assert.Nil(t, got) | ||||
| 	assert.False(t, ok) | ||||
|  | ||||
| 	// Should find object by other name | ||||
| 	got, ok = c.Get(context.Background(), testIndexName, "foo") | ||||
| 	assert.Equal(t, obj, got) | ||||
| 	assert.True(t, ok) | ||||
|  | ||||
| 	// Should find object by id | ||||
| 	got, ok = c.Get(context.Background(), testIndexID, "id") | ||||
| 	assert.Equal(t, obj, got) | ||||
| 	assert.True(t, ok) | ||||
| } | ||||
|  | ||||
| func Test_mapCache_Prune(t *testing.T) { | ||||
| 	c := NewCache[testIndex, string, *testObject](context.Background(), testIndices, cache.Config{ | ||||
| 		MaxAge:     time.Second, | ||||
| 		LastUseAge: time.Second / 4, | ||||
| 		Log: &logging.Config{ | ||||
| 			Level:     "debug", | ||||
| 			AddSource: true, | ||||
| 		}, | ||||
| 	}) | ||||
|  | ||||
| 	objects := []*testObject{ | ||||
| 		{ | ||||
| 			id:    "id1", | ||||
| 			names: []string{"foo", "bar"}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			id:    "id2", | ||||
| 			names: []string{"hello"}, | ||||
| 		}, | ||||
| 	} | ||||
| 	for _, obj := range objects { | ||||
| 		c.Set(context.Background(), obj) | ||||
| 	} | ||||
| 	// invalidate one entry | ||||
| 	err := c.Invalidate(context.Background(), testIndexName, "bar") | ||||
| 	require.NoError(t, err) | ||||
|  | ||||
| 	err = c.(cache.Pruner).Prune(context.Background()) | ||||
| 	require.NoError(t, err) | ||||
|  | ||||
| 	// Other object should still be found | ||||
| 	got, ok := c.Get(context.Background(), testIndexID, "id2") | ||||
| 	assert.Equal(t, objects[1], got) | ||||
| 	assert.True(t, ok) | ||||
| } | ||||
|  | ||||
| func Test_mapCache_Truncate(t *testing.T) { | ||||
| 	c := NewCache[testIndex, string, *testObject](context.Background(), testIndices, cache.Config{ | ||||
| 		MaxAge:     time.Second, | ||||
| 		LastUseAge: time.Second / 4, | ||||
| 		Log: &logging.Config{ | ||||
| 			Level:     "debug", | ||||
| 			AddSource: true, | ||||
| 		}, | ||||
| 	}) | ||||
| 	objects := []*testObject{ | ||||
| 		{ | ||||
| 			id:    "id1", | ||||
| 			names: []string{"foo", "bar"}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			id:    "id2", | ||||
| 			names: []string{"hello"}, | ||||
| 		}, | ||||
| 	} | ||||
| 	for _, obj := range objects { | ||||
| 		c.Set(context.Background(), obj) | ||||
| 	} | ||||
|  | ||||
| 	err := c.Truncate(context.Background()) | ||||
| 	require.NoError(t, err) | ||||
|  | ||||
| 	mc := c.(*mapCache[testIndex, string, *testObject]) | ||||
| 	for _, index := range mc.indexMap { | ||||
| 		index.mutex.RLock() | ||||
| 		assert.Len(t, index.entries, 0) | ||||
| 		index.mutex.RUnlock() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func Test_entry_isValid(t *testing.T) { | ||||
| 	type fields struct { | ||||
| 		created time.Time | ||||
| 		invalid bool | ||||
| 		lastUse time.Time | ||||
| 	} | ||||
| 	tests := []struct { | ||||
| 		name   string | ||||
| 		fields fields | ||||
| 		config *cache.Config | ||||
| 		want   bool | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name: "invalid", | ||||
| 			fields: fields{ | ||||
| 				created: time.Now(), | ||||
| 				invalid: true, | ||||
| 				lastUse: time.Now(), | ||||
| 			}, | ||||
| 			config: &cache.Config{ | ||||
| 				MaxAge:     time.Minute, | ||||
| 				LastUseAge: time.Second, | ||||
| 			}, | ||||
| 			want: false, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "max age exceeded", | ||||
| 			fields: fields{ | ||||
| 				created: time.Now().Add(-(time.Minute + time.Second)), | ||||
| 				invalid: false, | ||||
| 				lastUse: time.Now(), | ||||
| 			}, | ||||
| 			config: &cache.Config{ | ||||
| 				MaxAge:     time.Minute, | ||||
| 				LastUseAge: time.Second, | ||||
| 			}, | ||||
| 			want: false, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "max age disabled", | ||||
| 			fields: fields{ | ||||
| 				created: time.Now().Add(-(time.Minute + time.Second)), | ||||
| 				invalid: false, | ||||
| 				lastUse: time.Now(), | ||||
| 			}, | ||||
| 			config: &cache.Config{ | ||||
| 				LastUseAge: time.Second, | ||||
| 			}, | ||||
| 			want: true, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "last use age exceeded", | ||||
| 			fields: fields{ | ||||
| 				created: time.Now().Add(-(time.Minute / 2)), | ||||
| 				invalid: false, | ||||
| 				lastUse: time.Now().Add(-(time.Second * 2)), | ||||
| 			}, | ||||
| 			config: &cache.Config{ | ||||
| 				MaxAge:     time.Minute, | ||||
| 				LastUseAge: time.Second, | ||||
| 			}, | ||||
| 			want: false, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "last use age disabled", | ||||
| 			fields: fields{ | ||||
| 				created: time.Now().Add(-(time.Minute / 2)), | ||||
| 				invalid: false, | ||||
| 				lastUse: time.Now().Add(-(time.Second * 2)), | ||||
| 			}, | ||||
| 			config: &cache.Config{ | ||||
| 				MaxAge: time.Minute, | ||||
| 			}, | ||||
| 			want: true, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "valid", | ||||
| 			fields: fields{ | ||||
| 				created: time.Now(), | ||||
| 				invalid: false, | ||||
| 				lastUse: time.Now(), | ||||
| 			}, | ||||
| 			config: &cache.Config{ | ||||
| 				MaxAge:     time.Minute, | ||||
| 				LastUseAge: time.Second, | ||||
| 			}, | ||||
| 			want: true, | ||||
| 		}, | ||||
| 	} | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			e := &entry[any]{ | ||||
| 				created: tt.fields.created, | ||||
| 			} | ||||
| 			e.invalid.Store(tt.fields.invalid) | ||||
| 			e.lastUse.Store(tt.fields.lastUse.UnixMicro()) | ||||
| 			got := e.isValid(tt.config) | ||||
| 			assert.Equal(t, tt.want, got) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
		Reference in New Issue
	
	Block a user
	 Tim Möhlmann
					Tim Möhlmann