2021-08-09 08:01:20 +02:00
|
|
|
package command
|
|
|
|
|
|
|
|
import (
|
2025-07-04 18:12:59 +02:00
|
|
|
"bytes"
|
2021-08-09 08:01:20 +02:00
|
|
|
"context"
|
|
|
|
|
2022-04-27 01:01:45 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/domain"
|
|
|
|
"github.com/zitadel/zitadel/internal/eventstore"
|
|
|
|
"github.com/zitadel/zitadel/internal/repository/user"
|
2024-06-19 13:56:33 +03:00
|
|
|
"github.com/zitadel/zitadel/internal/telemetry/tracing"
|
2023-12-08 16:30:55 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/zerrors"
|
2021-08-09 08:01:20 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func (c *Commands) SetUserMetadata(ctx context.Context, metadata *domain.Metadata, userID, resourceOwner string) (_ *domain.Metadata, err error) {
|
2024-06-19 13:56:33 +03:00
|
|
|
ctx, span := tracing.NewSpan(ctx)
|
|
|
|
defer func() { span.EndWithError(err) }()
|
|
|
|
|
2025-07-04 18:12:59 +02:00
|
|
|
userResourceOwner, err := c.checkUserExists(ctx, userID, resourceOwner)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := c.checkPermissionUpdateUser(ctx, userResourceOwner, userID); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
setMetadata, err := c.getUserMetadataModelByID(ctx, userID, userResourceOwner, metadata.Key)
|
2021-08-09 08:01:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
userAgg := UserAggregateFromWriteModel(&setMetadata.WriteModel)
|
2025-07-04 18:12:59 +02:00
|
|
|
// return if no change in the metadata
|
|
|
|
if bytes.Equal(setMetadata.Value, metadata.Value) {
|
|
|
|
return writeModelToUserMetadata(setMetadata), nil
|
|
|
|
}
|
|
|
|
|
2021-08-09 08:01:20 +02:00
|
|
|
event, err := c.setUserMetadata(ctx, userAgg, metadata)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-01-03 09:19:07 +01:00
|
|
|
pushedEvents, err := c.eventstore.Push(ctx, event)
|
2021-08-09 08:01:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = AppendAndReduce(setMetadata, pushedEvents...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return writeModelToUserMetadata(setMetadata), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Commands) BulkSetUserMetadata(ctx context.Context, userID, resourceOwner string, metadatas ...*domain.Metadata) (_ *domain.ObjectDetails, err error) {
|
|
|
|
if len(metadatas) == 0 {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowPreconditionFailed(nil, "META-9mm2d", "Errors.Metadata.NoData")
|
2021-08-09 08:01:20 +02:00
|
|
|
}
|
2025-07-04 18:12:59 +02:00
|
|
|
userResourceOwner, err := c.checkUserExists(ctx, userID, resourceOwner)
|
2021-08-09 08:01:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2025-07-04 18:12:59 +02:00
|
|
|
if err := c.checkPermissionUpdateUser(ctx, userResourceOwner, userID); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
events := make([]eventstore.Command, 0)
|
|
|
|
setMetadata, err := c.getUserMetadataListModelByID(ctx, userID, userResourceOwner)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-08-09 08:01:20 +02:00
|
|
|
userAgg := UserAggregateFromWriteModel(&setMetadata.WriteModel)
|
2025-07-04 18:12:59 +02:00
|
|
|
for _, data := range metadatas {
|
|
|
|
// if no change to metadata no event has to be pushed
|
|
|
|
if existingValue, ok := setMetadata.metadataList[data.Key]; ok && bytes.Equal(existingValue, data.Value) {
|
|
|
|
continue
|
|
|
|
}
|
2021-08-09 08:01:20 +02:00
|
|
|
event, err := c.setUserMetadata(ctx, userAgg, data)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2025-07-04 18:12:59 +02:00
|
|
|
events = append(events, event)
|
|
|
|
}
|
|
|
|
// no changes for the metadata
|
|
|
|
if len(events) == 0 {
|
|
|
|
return writeModelToObjectDetails(&setMetadata.WriteModel), nil
|
2021-08-09 08:01:20 +02:00
|
|
|
}
|
|
|
|
|
2022-01-03 09:19:07 +01:00
|
|
|
pushedEvents, err := c.eventstore.Push(ctx, events...)
|
2021-08-09 08:01:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = AppendAndReduce(setMetadata, pushedEvents...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return writeModelToObjectDetails(&setMetadata.WriteModel), nil
|
|
|
|
}
|
|
|
|
|
2022-01-03 09:19:07 +01:00
|
|
|
func (c *Commands) setUserMetadata(ctx context.Context, userAgg *eventstore.Aggregate, metadata *domain.Metadata) (command eventstore.Command, err error) {
|
2021-08-09 08:01:20 +02:00
|
|
|
if !metadata.IsValid() {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgument(nil, "META-2m00f", "Errors.Metadata.Invalid")
|
2021-08-09 08:01:20 +02:00
|
|
|
}
|
|
|
|
return user.NewMetadataSetEvent(
|
|
|
|
ctx,
|
|
|
|
userAgg,
|
|
|
|
metadata.Key,
|
|
|
|
metadata.Value,
|
|
|
|
), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Commands) RemoveUserMetadata(ctx context.Context, metadataKey, userID, resourceOwner string) (_ *domain.ObjectDetails, err error) {
|
|
|
|
if metadataKey == "" {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgument(nil, "META-2n0fs", "Errors.Metadata.Invalid")
|
2021-08-09 08:01:20 +02:00
|
|
|
}
|
2025-07-04 18:12:59 +02:00
|
|
|
userResourceOwner, err := c.checkUserExists(ctx, userID, resourceOwner)
|
2021-08-09 08:01:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2025-07-04 18:12:59 +02:00
|
|
|
|
|
|
|
if err := c.checkPermissionUpdateUser(ctx, userResourceOwner, userID); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
removeMetadata, err := c.getUserMetadataModelByID(ctx, userID, userResourceOwner, metadataKey)
|
2021-08-09 08:01:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if !removeMetadata.State.Exists() {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowNotFound(nil, "META-ncnw3", "Errors.Metadata.NotFound")
|
2021-08-09 08:01:20 +02:00
|
|
|
}
|
|
|
|
userAgg := UserAggregateFromWriteModel(&removeMetadata.WriteModel)
|
|
|
|
event, err := c.removeUserMetadata(ctx, userAgg, metadataKey)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-01-03 09:19:07 +01:00
|
|
|
pushedEvents, err := c.eventstore.Push(ctx, event)
|
2021-08-09 08:01:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = AppendAndReduce(removeMetadata, pushedEvents...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return writeModelToObjectDetails(&removeMetadata.WriteModel), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Commands) BulkRemoveUserMetadata(ctx context.Context, userID, resourceOwner string, metadataKeys ...string) (_ *domain.ObjectDetails, err error) {
|
|
|
|
if len(metadataKeys) == 0 {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowPreconditionFailed(nil, "META-9mm2d", "Errors.Metadata.NoData")
|
2021-08-09 08:01:20 +02:00
|
|
|
}
|
2025-07-04 18:12:59 +02:00
|
|
|
userResourceOwner, err := c.checkUserExists(ctx, userID, resourceOwner)
|
2021-08-09 08:01:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2025-07-04 18:12:59 +02:00
|
|
|
if err := c.checkPermissionUpdateUser(ctx, userResourceOwner, userID); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-01-03 09:19:07 +01:00
|
|
|
events := make([]eventstore.Command, len(metadataKeys))
|
2025-07-04 18:12:59 +02:00
|
|
|
removeMetadata, err := c.getUserMetadataListModelByID(ctx, userID, userResourceOwner)
|
2021-08-09 08:01:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
userAgg := UserAggregateFromWriteModel(&removeMetadata.WriteModel)
|
|
|
|
for i, key := range metadataKeys {
|
|
|
|
if key == "" {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgument(nil, "COMMAND-m29ds", "Errors.Metadata.Invalid")
|
2021-08-09 08:01:20 +02:00
|
|
|
}
|
|
|
|
if _, found := removeMetadata.metadataList[key]; !found {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowNotFound(nil, "META-2nnds", "Errors.Metadata.KeyNotExisting")
|
2021-08-09 08:01:20 +02:00
|
|
|
}
|
|
|
|
event, err := c.removeUserMetadata(ctx, userAgg, key)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
events[i] = event
|
|
|
|
}
|
|
|
|
|
2022-01-03 09:19:07 +01:00
|
|
|
pushedEvents, err := c.eventstore.Push(ctx, events...)
|
2021-08-09 08:01:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = AppendAndReduce(removeMetadata, pushedEvents...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return writeModelToObjectDetails(&removeMetadata.WriteModel), nil
|
|
|
|
}
|
|
|
|
|
2022-01-03 09:19:07 +01:00
|
|
|
func (c *Commands) removeUserMetadata(ctx context.Context, userAgg *eventstore.Aggregate, metadataKey string) (command eventstore.Command, err error) {
|
|
|
|
command = user.NewMetadataRemovedEvent(
|
2021-08-09 08:01:20 +02:00
|
|
|
ctx,
|
|
|
|
userAgg,
|
|
|
|
metadataKey,
|
|
|
|
)
|
2022-01-03 09:19:07 +01:00
|
|
|
return command, nil
|
2021-08-09 08:01:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Commands) getUserMetadataModelByID(ctx context.Context, userID, resourceOwner, key string) (*UserMetadataWriteModel, error) {
|
|
|
|
userMetadataWriteModel := NewUserMetadataWriteModel(userID, resourceOwner, key)
|
|
|
|
err := c.eventstore.FilterToQueryReducer(ctx, userMetadataWriteModel)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return userMetadataWriteModel, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Commands) getUserMetadataListModelByID(ctx context.Context, userID, resourceOwner string) (*UserMetadataListWriteModel, error) {
|
|
|
|
userMetadataWriteModel := NewUserMetadataListWriteModel(userID, resourceOwner)
|
|
|
|
err := c.eventstore.FilterToQueryReducer(ctx, userMetadataWriteModel)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return userMetadataWriteModel, nil
|
|
|
|
}
|