2020-09-30 10:00:05 +02:00
|
|
|
package repository
|
2020-09-24 08:52:10 +02:00
|
|
|
|
2022-05-19 13:44:16 +02:00
|
|
|
import (
|
|
|
|
"database/sql"
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/database"
|
|
|
|
"github.com/zitadel/zitadel/internal/eventstore"
|
2023-12-08 16:30:55 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/zerrors"
|
2022-05-19 13:44:16 +02:00
|
|
|
)
|
2020-09-30 10:00:05 +02:00
|
|
|
|
2022-10-31 13:03:23 +00:00
|
|
|
// SearchQuery defines the which and how data are queried
|
2020-09-30 10:00:05 +02:00
|
|
|
type SearchQuery struct {
|
2023-10-19 12:19:10 +02:00
|
|
|
Columns eventstore.Columns
|
2020-09-30 10:00:05 +02:00
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
SubQueries [][]*Filter
|
|
|
|
Tx *sql.Tx
|
|
|
|
AllowTimeTravel bool
|
|
|
|
AwaitOpenTransactions bool
|
|
|
|
Limit uint64
|
2023-12-21 11:40:51 +01:00
|
|
|
Offset uint32
|
2023-10-19 12:19:10 +02:00
|
|
|
Desc bool
|
2020-09-24 08:52:10 +02:00
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
InstanceID *Filter
|
2024-01-17 11:16:48 +01:00
|
|
|
InstanceIDs *Filter
|
2023-10-19 12:19:10 +02:00
|
|
|
ExcludedInstances *Filter
|
|
|
|
Creator *Filter
|
|
|
|
Owner *Filter
|
|
|
|
Position *Filter
|
|
|
|
Sequence *Filter
|
2023-11-03 15:52:48 +01:00
|
|
|
CreatedAfter *Filter
|
|
|
|
CreatedBefore *Filter
|
2020-10-06 21:28:09 +02:00
|
|
|
}
|
|
|
|
|
2022-10-31 13:03:23 +00:00
|
|
|
// Filter represents all fields needed to compare a field of an event with a value
|
2020-09-24 08:52:10 +02:00
|
|
|
type Filter struct {
|
2020-10-05 20:39:36 +02:00
|
|
|
Field Field
|
|
|
|
Value interface{}
|
|
|
|
Operation Operation
|
2020-09-24 08:52:10 +02:00
|
|
|
}
|
|
|
|
|
2022-10-31 13:03:23 +00:00
|
|
|
// Operation defines how fields are compared
|
2020-09-24 08:52:10 +02:00
|
|
|
type Operation int32
|
|
|
|
|
|
|
|
const (
|
2020-10-06 21:28:09 +02:00
|
|
|
// OperationEquals compares two values for equality
|
|
|
|
OperationEquals Operation = iota + 1
|
|
|
|
// OperationGreater compares if the given values is greater than the stored one
|
|
|
|
OperationGreater
|
|
|
|
// OperationLess compares if the given values is less than the stored one
|
|
|
|
OperationLess
|
|
|
|
//OperationIn checks if a stored value matches one of the passed value list
|
|
|
|
OperationIn
|
2020-11-23 19:31:12 +01:00
|
|
|
//OperationJSONContains checks if a stored value matches the given json
|
|
|
|
OperationJSONContains
|
2022-04-19 08:26:12 +02:00
|
|
|
//OperationNotIn checks if a stored value does not match one of the passed value list
|
|
|
|
OperationNotIn
|
2020-10-06 21:28:09 +02:00
|
|
|
|
|
|
|
operationCount
|
2020-09-24 08:52:10 +02:00
|
|
|
)
|
|
|
|
|
2022-10-31 13:03:23 +00:00
|
|
|
// Field is the representation of a field from the event
|
2020-09-24 08:52:10 +02:00
|
|
|
type Field int32
|
|
|
|
|
|
|
|
const (
|
2020-10-06 21:28:09 +02:00
|
|
|
//FieldAggregateType represents the aggregate type field
|
|
|
|
FieldAggregateType Field = iota + 1
|
|
|
|
//FieldAggregateID represents the aggregate id field
|
|
|
|
FieldAggregateID
|
|
|
|
//FieldSequence represents the sequence field
|
|
|
|
FieldSequence
|
|
|
|
//FieldResourceOwner represents the resource owner field
|
|
|
|
FieldResourceOwner
|
2022-03-23 09:02:39 +01:00
|
|
|
//FieldInstanceID represents the instance id field
|
|
|
|
FieldInstanceID
|
2020-10-06 21:28:09 +02:00
|
|
|
//FieldEditorService represents the editor service field
|
|
|
|
FieldEditorService
|
|
|
|
//FieldEditorUser represents the editor user field
|
|
|
|
FieldEditorUser
|
|
|
|
//FieldEventType represents the event type field
|
|
|
|
FieldEventType
|
2020-11-23 19:31:12 +01:00
|
|
|
//FieldEventData represents the event data field
|
|
|
|
FieldEventData
|
2022-09-02 16:05:13 +02:00
|
|
|
//FieldCreationDate represents the creation date field
|
|
|
|
FieldCreationDate
|
2023-10-19 12:19:10 +02:00
|
|
|
// FieldPosition represents the field of the global sequence
|
|
|
|
FieldPosition
|
2020-10-06 21:28:09 +02:00
|
|
|
|
|
|
|
fieldCount
|
2020-09-24 08:52:10 +02:00
|
|
|
)
|
|
|
|
|
2022-10-31 13:03:23 +00:00
|
|
|
// NewFilter is used in tests. Use searchQuery.*Filter() instead
|
2020-09-24 08:52:10 +02:00
|
|
|
func NewFilter(field Field, value interface{}, operation Operation) *Filter {
|
|
|
|
return &Filter{
|
2020-10-05 20:39:36 +02:00
|
|
|
Field: field,
|
|
|
|
Value: value,
|
|
|
|
Operation: operation,
|
2020-09-24 08:52:10 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-31 13:03:23 +00:00
|
|
|
// Validate checks if the fields of the filter have valid values
|
2020-09-24 08:52:10 +02:00
|
|
|
func (f *Filter) Validate() error {
|
|
|
|
if f == nil {
|
2023-12-08 16:30:55 +02:00
|
|
|
return zerrors.ThrowPreconditionFailed(nil, "REPO-z6KcG", "filter is nil")
|
2020-09-24 08:52:10 +02:00
|
|
|
}
|
2020-10-06 21:28:09 +02:00
|
|
|
if f.Field <= 0 || f.Field >= fieldCount {
|
2023-12-08 16:30:55 +02:00
|
|
|
return zerrors.ThrowPreconditionFailed(nil, "REPO-zw62U", "field not definded")
|
2020-09-24 08:52:10 +02:00
|
|
|
}
|
2020-10-05 20:39:36 +02:00
|
|
|
if f.Value == nil {
|
2023-12-08 16:30:55 +02:00
|
|
|
return zerrors.ThrowPreconditionFailed(nil, "REPO-GJ9ct", "no value definded")
|
2020-09-24 08:52:10 +02:00
|
|
|
}
|
2020-10-06 21:28:09 +02:00
|
|
|
if f.Operation <= 0 || f.Operation >= operationCount {
|
2023-12-08 16:30:55 +02:00
|
|
|
return zerrors.ThrowPreconditionFailed(nil, "REPO-RrQTy", "operation not definded")
|
2020-09-24 08:52:10 +02:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2023-10-19 12:19:10 +02:00
|
|
|
|
|
|
|
func QueryFromBuilder(builder *eventstore.SearchQueryBuilder) (*SearchQuery, error) {
|
|
|
|
if builder == nil ||
|
|
|
|
builder.GetColumns().Validate() != nil {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowPreconditionFailed(nil, "MODEL-4m9gs", "builder invalid")
|
2023-10-19 12:19:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
query := &SearchQuery{
|
|
|
|
Columns: builder.GetColumns(),
|
|
|
|
Limit: builder.GetLimit(),
|
2023-12-01 13:25:41 +01:00
|
|
|
Offset: builder.GetOffset(),
|
2023-10-19 12:19:10 +02:00
|
|
|
Desc: builder.GetDesc(),
|
|
|
|
Tx: builder.GetTx(),
|
|
|
|
AllowTimeTravel: builder.GetAllowTimeTravel(),
|
|
|
|
AwaitOpenTransactions: builder.GetAwaitOpenTransactions(),
|
2023-12-01 13:25:41 +01:00
|
|
|
SubQueries: make([][]*Filter, len(builder.GetQueries())),
|
2023-10-19 12:19:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, f := range []func(builder *eventstore.SearchQueryBuilder, query *SearchQuery) *Filter{
|
|
|
|
instanceIDFilter,
|
2024-01-17 11:16:48 +01:00
|
|
|
instanceIDsFilter,
|
2023-10-19 12:19:10 +02:00
|
|
|
editorUserFilter,
|
|
|
|
resourceOwnerFilter,
|
|
|
|
positionAfterFilter,
|
|
|
|
eventSequenceGreaterFilter,
|
|
|
|
creationDateAfterFilter,
|
2023-11-03 15:52:48 +01:00
|
|
|
creationDateBeforeFilter,
|
2023-10-19 12:19:10 +02:00
|
|
|
} {
|
|
|
|
filter := f(builder, query)
|
|
|
|
if filter == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err := filter.Validate(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, q := range builder.GetQueries() {
|
|
|
|
for _, f := range []func(query *eventstore.SearchQuery) *Filter{
|
|
|
|
aggregateTypeFilter,
|
|
|
|
aggregateIDFilter,
|
|
|
|
eventTypeFilter,
|
|
|
|
eventDataFilter,
|
perf(oidc): nest position clause for session terminated query (#8738)
# Which Problems Are Solved
Optimize the query that checks for terminated sessions in the access
token verifier. The verifier is used in auth middleware, userinfo and
introspection.
# How the Problems Are Solved
The previous implementation built a query for certain events and then
appended a single `PositionAfter` clause. This caused the postgreSQL
planner to use indexes only for the instance ID, aggregate IDs,
aggregate types and event types. Followed by an expensive sequential
scan for the position. This resulting in internal over-fetching of rows
before the final filter was applied.
![Screenshot_20241007_105803](https://github.com/user-attachments/assets/f2d91976-be87-428b-b604-a211399b821c)
Furthermore, the query was searching for events which are not always
applicable. For example, there was always a session ID search and if
there was a user ID, we would also search for a browser fingerprint in
event payload (expensive). Even if those argument string would be empty.
This PR changes:
1. Nest the position query, so that a full `instance_id, aggregate_id,
aggregate_type, event_type, "position"` index can be matched.
2. Redefine the `es_wm` index to include the `position` column.
3. Only search for events for the IDs that actually have a value. Do not
search (noop) if none of session ID, user ID or fingerpint ID are set.
New query plan:
![Screenshot_20241007_110648](https://github.com/user-attachments/assets/c3234c33-1b76-4b33-a4a9-796f69f3d775)
# Additional Changes
- cleanup how we load multi-statement migrations and make that a bit
more reusable.
# Additional Context
- Related to https://github.com/zitadel/zitadel/issues/7639
2024-10-07 15:49:55 +03:00
|
|
|
eventPositionAfterFilter,
|
2023-10-19 12:19:10 +02:00
|
|
|
} {
|
|
|
|
filter := f(q)
|
|
|
|
if filter == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err := filter.Validate(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
query.SubQueries[i] = append(query.SubQueries[i], filter)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return query, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func eventSequenceGreaterFilter(builder *eventstore.SearchQueryBuilder, query *SearchQuery) *Filter {
|
|
|
|
if builder.GetEventSequenceGreater() == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
sortOrder := OperationGreater
|
|
|
|
if builder.GetDesc() {
|
|
|
|
sortOrder = OperationLess
|
|
|
|
}
|
|
|
|
query.Sequence = NewFilter(FieldSequence, builder.GetEventSequenceGreater(), sortOrder)
|
|
|
|
return query.Sequence
|
|
|
|
}
|
|
|
|
|
|
|
|
func creationDateAfterFilter(builder *eventstore.SearchQueryBuilder, query *SearchQuery) *Filter {
|
|
|
|
if builder.GetCreationDateAfter().IsZero() {
|
|
|
|
return nil
|
|
|
|
}
|
2023-11-03 15:52:48 +01:00
|
|
|
query.CreatedAfter = NewFilter(FieldCreationDate, builder.GetCreationDateAfter(), OperationGreater)
|
|
|
|
return query.CreatedAfter
|
|
|
|
}
|
|
|
|
|
|
|
|
func creationDateBeforeFilter(builder *eventstore.SearchQueryBuilder, query *SearchQuery) *Filter {
|
|
|
|
if builder.GetCreationDateBefore().IsZero() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
query.CreatedBefore = NewFilter(FieldCreationDate, builder.GetCreationDateBefore(), OperationLess)
|
|
|
|
return query.CreatedBefore
|
2023-10-19 12:19:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func resourceOwnerFilter(builder *eventstore.SearchQueryBuilder, query *SearchQuery) *Filter {
|
|
|
|
if builder.GetResourceOwner() == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
query.Owner = NewFilter(FieldResourceOwner, builder.GetResourceOwner(), OperationEquals)
|
|
|
|
return query.Owner
|
|
|
|
}
|
|
|
|
|
|
|
|
func editorUserFilter(builder *eventstore.SearchQueryBuilder, query *SearchQuery) *Filter {
|
|
|
|
if builder.GetEditorUser() == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
query.Creator = NewFilter(FieldEditorUser, builder.GetEditorUser(), OperationEquals)
|
|
|
|
return query.Creator
|
|
|
|
}
|
|
|
|
|
|
|
|
func instanceIDFilter(builder *eventstore.SearchQueryBuilder, query *SearchQuery) *Filter {
|
|
|
|
if builder.GetInstanceID() == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
query.InstanceID = NewFilter(FieldInstanceID, *builder.GetInstanceID(), OperationEquals)
|
|
|
|
return query.InstanceID
|
|
|
|
}
|
|
|
|
|
2024-01-17 11:16:48 +01:00
|
|
|
func instanceIDsFilter(builder *eventstore.SearchQueryBuilder, query *SearchQuery) *Filter {
|
|
|
|
if builder.GetInstanceIDs() == nil {
|
|
|
|
return nil
|
|
|
|
}
|
2024-03-27 14:48:22 +01:00
|
|
|
query.InstanceIDs = NewFilter(FieldInstanceID, database.TextArray[string](builder.GetInstanceIDs()), OperationIn)
|
2024-01-17 11:16:48 +01:00
|
|
|
return query.InstanceIDs
|
|
|
|
}
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
func positionAfterFilter(builder *eventstore.SearchQueryBuilder, query *SearchQuery) *Filter {
|
2024-09-24 19:43:29 +03:00
|
|
|
if builder.GetPositionAfter() == 0 {
|
2023-10-19 12:19:10 +02:00
|
|
|
return nil
|
|
|
|
}
|
2024-09-24 19:43:29 +03:00
|
|
|
query.Position = NewFilter(FieldPosition, builder.GetPositionAfter(), OperationGreater)
|
2023-10-19 12:19:10 +02:00
|
|
|
return query.Position
|
|
|
|
}
|
|
|
|
|
|
|
|
func aggregateIDFilter(query *eventstore.SearchQuery) *Filter {
|
|
|
|
if len(query.GetAggregateIDs()) < 1 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if len(query.GetAggregateIDs()) == 1 {
|
|
|
|
return NewFilter(FieldAggregateID, query.GetAggregateIDs()[0], OperationEquals)
|
|
|
|
}
|
|
|
|
return NewFilter(FieldAggregateID, database.TextArray[string](query.GetAggregateIDs()), OperationIn)
|
|
|
|
}
|
|
|
|
|
|
|
|
func eventTypeFilter(query *eventstore.SearchQuery) *Filter {
|
|
|
|
if len(query.GetEventTypes()) < 1 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if len(query.GetEventTypes()) == 1 {
|
|
|
|
return NewFilter(FieldEventType, query.GetEventTypes()[0], OperationEquals)
|
|
|
|
}
|
2024-03-27 14:48:22 +01:00
|
|
|
return NewFilter(FieldEventType, database.TextArray[eventstore.EventType](query.GetEventTypes()), OperationIn)
|
2023-10-19 12:19:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func aggregateTypeFilter(query *eventstore.SearchQuery) *Filter {
|
|
|
|
if len(query.GetAggregateTypes()) < 1 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if len(query.GetAggregateTypes()) == 1 {
|
|
|
|
return NewFilter(FieldAggregateType, query.GetAggregateTypes()[0], OperationEquals)
|
|
|
|
}
|
2024-03-27 14:48:22 +01:00
|
|
|
return NewFilter(FieldAggregateType, database.TextArray[eventstore.AggregateType](query.GetAggregateTypes()), OperationIn)
|
2023-10-19 12:19:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func eventDataFilter(query *eventstore.SearchQuery) *Filter {
|
|
|
|
if len(query.GetEventData()) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return NewFilter(FieldEventData, query.GetEventData(), OperationJSONContains)
|
|
|
|
}
|
perf(oidc): nest position clause for session terminated query (#8738)
# Which Problems Are Solved
Optimize the query that checks for terminated sessions in the access
token verifier. The verifier is used in auth middleware, userinfo and
introspection.
# How the Problems Are Solved
The previous implementation built a query for certain events and then
appended a single `PositionAfter` clause. This caused the postgreSQL
planner to use indexes only for the instance ID, aggregate IDs,
aggregate types and event types. Followed by an expensive sequential
scan for the position. This resulting in internal over-fetching of rows
before the final filter was applied.
![Screenshot_20241007_105803](https://github.com/user-attachments/assets/f2d91976-be87-428b-b604-a211399b821c)
Furthermore, the query was searching for events which are not always
applicable. For example, there was always a session ID search and if
there was a user ID, we would also search for a browser fingerprint in
event payload (expensive). Even if those argument string would be empty.
This PR changes:
1. Nest the position query, so that a full `instance_id, aggregate_id,
aggregate_type, event_type, "position"` index can be matched.
2. Redefine the `es_wm` index to include the `position` column.
3. Only search for events for the IDs that actually have a value. Do not
search (noop) if none of session ID, user ID or fingerpint ID are set.
New query plan:
![Screenshot_20241007_110648](https://github.com/user-attachments/assets/c3234c33-1b76-4b33-a4a9-796f69f3d775)
# Additional Changes
- cleanup how we load multi-statement migrations and make that a bit
more reusable.
# Additional Context
- Related to https://github.com/zitadel/zitadel/issues/7639
2024-10-07 15:49:55 +03:00
|
|
|
|
|
|
|
func eventPositionAfterFilter(query *eventstore.SearchQuery) *Filter {
|
|
|
|
if pos := query.GetPositionAfter(); pos != 0 {
|
|
|
|
return NewFilter(FieldPosition, pos, OperationGreater)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|