mirror of
https://github.com/zitadel/zitadel.git
synced 2025-08-11 21:17:32 +00:00
feat(eventstore): increase parallel write capabilities (#5940)
This implementation increases parallel write capabilities of the eventstore. Please have a look at the technical advisories: [05](https://zitadel.com/docs/support/advisory/a10005) and [06](https://zitadel.com/docs/support/advisory/a10006). The implementation of eventstore.push is rewritten and stored events are migrated to a new table `eventstore.events2`. If you are using cockroach: make sure that the database user of ZITADEL has `VIEWACTIVITY` grant. This is used to query events.
This commit is contained in:
293
internal/eventstore/v3/sequence_test.go
Normal file
293
internal/eventstore/v3/sequence_test.go
Normal file
@@ -0,0 +1,293 @@
|
||||
package eventstore
|
||||
|
||||
import (
|
||||
"context"
|
||||
_ "embed"
|
||||
"reflect"
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/assert"
|
||||
|
||||
"github.com/zitadel/zitadel/internal/api/authz"
|
||||
"github.com/zitadel/zitadel/internal/eventstore"
|
||||
)
|
||||
|
||||
func Test_searchSequence(t *testing.T) {
|
||||
sequence := &latestSequence{
|
||||
aggregate: mockAggregate("V3-p1BWC"),
|
||||
sequence: 1,
|
||||
}
|
||||
type args struct {
|
||||
sequences []*latestSequence
|
||||
aggregateType eventstore.AggregateType
|
||||
aggregateID string
|
||||
instanceID string
|
||||
}
|
||||
tests := []struct {
|
||||
name string
|
||||
args args
|
||||
want *latestSequence
|
||||
}{
|
||||
{
|
||||
name: "type missmatch",
|
||||
args: args{
|
||||
sequences: []*latestSequence{
|
||||
sequence,
|
||||
},
|
||||
aggregateType: "wrong",
|
||||
aggregateID: "V3-p1BWC",
|
||||
instanceID: "instance",
|
||||
},
|
||||
want: nil,
|
||||
},
|
||||
{
|
||||
name: "id missmatch",
|
||||
args: args{
|
||||
sequences: []*latestSequence{
|
||||
sequence,
|
||||
},
|
||||
aggregateType: "type",
|
||||
aggregateID: "wrong",
|
||||
instanceID: "instance",
|
||||
},
|
||||
want: nil,
|
||||
},
|
||||
{
|
||||
name: "instance missmatch",
|
||||
args: args{
|
||||
sequences: []*latestSequence{
|
||||
sequence,
|
||||
},
|
||||
aggregateType: "type",
|
||||
aggregateID: "V3-p1BWC",
|
||||
instanceID: "wrong",
|
||||
},
|
||||
want: nil,
|
||||
},
|
||||
{
|
||||
name: "match",
|
||||
args: args{
|
||||
sequences: []*latestSequence{
|
||||
sequence,
|
||||
},
|
||||
aggregateType: "type",
|
||||
aggregateID: "V3-p1BWC",
|
||||
instanceID: "instance",
|
||||
},
|
||||
want: sequence,
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
if got := searchSequence(tt.args.sequences, tt.args.aggregateType, tt.args.aggregateID, tt.args.instanceID); !reflect.DeepEqual(got, tt.want) {
|
||||
t.Errorf("searchSequence() = %v, want %v", got, tt.want)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func Test_commandsToSequences(t *testing.T) {
|
||||
aggregate := mockAggregate("V3-MKHTF")
|
||||
type args struct {
|
||||
ctx context.Context
|
||||
commands []eventstore.Command
|
||||
}
|
||||
tests := []struct {
|
||||
name string
|
||||
args args
|
||||
want []*latestSequence
|
||||
}{
|
||||
{
|
||||
name: "no command",
|
||||
args: args{
|
||||
ctx: context.Background(),
|
||||
commands: []eventstore.Command{},
|
||||
},
|
||||
want: []*latestSequence{},
|
||||
},
|
||||
{
|
||||
name: "one command",
|
||||
args: args{
|
||||
ctx: context.Background(),
|
||||
commands: []eventstore.Command{
|
||||
&mockCommand{
|
||||
aggregate: aggregate,
|
||||
},
|
||||
},
|
||||
},
|
||||
want: []*latestSequence{
|
||||
{
|
||||
aggregate: aggregate,
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "two commands same aggregate",
|
||||
args: args{
|
||||
ctx: context.Background(),
|
||||
commands: []eventstore.Command{
|
||||
&mockCommand{
|
||||
aggregate: aggregate,
|
||||
},
|
||||
&mockCommand{
|
||||
aggregate: aggregate,
|
||||
},
|
||||
},
|
||||
},
|
||||
want: []*latestSequence{
|
||||
{
|
||||
aggregate: aggregate,
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "two commands different aggregates",
|
||||
args: args{
|
||||
ctx: context.Background(),
|
||||
commands: []eventstore.Command{
|
||||
&mockCommand{
|
||||
aggregate: aggregate,
|
||||
},
|
||||
&mockCommand{
|
||||
aggregate: mockAggregate("V3-cZkCy"),
|
||||
},
|
||||
},
|
||||
},
|
||||
want: []*latestSequence{
|
||||
{
|
||||
aggregate: aggregate,
|
||||
},
|
||||
{
|
||||
aggregate: mockAggregate("V3-cZkCy"),
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "instance set in command",
|
||||
args: args{
|
||||
ctx: authz.WithInstanceID(context.Background(), "V3-ANV4p"),
|
||||
commands: []eventstore.Command{
|
||||
&mockCommand{
|
||||
aggregate: &eventstore.Aggregate{
|
||||
ID: "V3-bF0Sa",
|
||||
Type: "type",
|
||||
ResourceOwner: "to",
|
||||
InstanceID: "instance",
|
||||
Version: "v1",
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
want: []*latestSequence{
|
||||
{
|
||||
aggregate: &eventstore.Aggregate{
|
||||
ID: "V3-bF0Sa",
|
||||
Type: "type",
|
||||
ResourceOwner: "to",
|
||||
InstanceID: "instance",
|
||||
Version: "v1",
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "instance from context",
|
||||
args: args{
|
||||
ctx: authz.WithInstanceID(context.Background(), "V3-ANV4p"),
|
||||
commands: []eventstore.Command{
|
||||
&mockCommand{
|
||||
aggregate: &eventstore.Aggregate{
|
||||
ID: "V3-bF0Sa",
|
||||
Type: "type",
|
||||
ResourceOwner: "to",
|
||||
Version: "v1",
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
want: []*latestSequence{
|
||||
{
|
||||
aggregate: &eventstore.Aggregate{
|
||||
ID: "V3-bF0Sa",
|
||||
Type: "type",
|
||||
ResourceOwner: "to",
|
||||
InstanceID: "V3-ANV4p",
|
||||
Version: "v1",
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
got := commandsToSequences(tt.args.ctx, tt.args.commands)
|
||||
assert.ElementsMatch(t, tt.want, got)
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func Test_sequencesToSql(t *testing.T) {
|
||||
tests := []struct {
|
||||
name string
|
||||
arg []*latestSequence
|
||||
wantConditions []string
|
||||
wantArgs []any
|
||||
}{
|
||||
{
|
||||
name: "no sequence",
|
||||
arg: []*latestSequence{},
|
||||
wantConditions: []string{},
|
||||
wantArgs: []any{},
|
||||
},
|
||||
{
|
||||
name: "one",
|
||||
arg: []*latestSequence{
|
||||
{
|
||||
aggregate: mockAggregate("V3-SbpGB"),
|
||||
},
|
||||
},
|
||||
wantConditions: []string{
|
||||
"(instance_id = $1 AND aggregate_type = $2 AND aggregate_id = $3)",
|
||||
},
|
||||
wantArgs: []any{
|
||||
"instance",
|
||||
eventstore.AggregateType("type"),
|
||||
"V3-SbpGB",
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "multiple",
|
||||
arg: []*latestSequence{
|
||||
{
|
||||
aggregate: mockAggregate("V3-SbpGB"),
|
||||
},
|
||||
{
|
||||
aggregate: mockAggregate("V3-0X3yt"),
|
||||
},
|
||||
},
|
||||
wantConditions: []string{
|
||||
"(instance_id = $1 AND aggregate_type = $2 AND aggregate_id = $3)",
|
||||
"(instance_id = $4 AND aggregate_type = $5 AND aggregate_id = $6)",
|
||||
},
|
||||
wantArgs: []any{
|
||||
"instance",
|
||||
eventstore.AggregateType("type"),
|
||||
"V3-SbpGB",
|
||||
"instance",
|
||||
eventstore.AggregateType("type"),
|
||||
"V3-0X3yt",
|
||||
},
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
gotConditions, gotArgs := sequencesToSql(tt.arg)
|
||||
if !reflect.DeepEqual(gotConditions, tt.wantConditions) {
|
||||
t.Errorf("sequencesToSql() gotConditions = %v, want %v", gotConditions, tt.wantConditions)
|
||||
}
|
||||
if !reflect.DeepEqual(gotArgs, tt.wantArgs) {
|
||||
t.Errorf("sequencesToSql() gotArgs = %v, want %v", gotArgs, tt.wantArgs)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user