mirror of
https://github.com/zitadel/zitadel.git
synced 2025-01-11 00:34:04 +00:00
5fdad7b8f4
# Which Problems Are Solved Users are not yet able to update their information an status in user API v3. # How the Problems Are Solved Add endpoints and functionality to update users and their status in user API v3. # Additional Changes Aggregate_type and event_types are updated with "userschema" to avoid conflicts with old events. # Additional Context closes #7898
195 lines
4.9 KiB
Go
195 lines
4.9 KiB
Go
package command
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
|
|
"github.com/zitadel/zitadel/internal/domain"
|
|
"github.com/zitadel/zitadel/internal/eventstore"
|
|
"github.com/zitadel/zitadel/internal/repository/user/schemauser"
|
|
)
|
|
|
|
type UserV3WriteModel struct {
|
|
eventstore.WriteModel
|
|
|
|
PhoneWM bool
|
|
EmailWM bool
|
|
DataWM bool
|
|
|
|
SchemaID string
|
|
SchemaRevision uint64
|
|
|
|
Email string
|
|
IsEmailVerified bool
|
|
EmailVerifiedFailedCount int
|
|
Phone string
|
|
IsPhoneVerified bool
|
|
PhoneVerifiedFailedCount int
|
|
|
|
Data json.RawMessage
|
|
|
|
Locked bool
|
|
State domain.UserState
|
|
}
|
|
|
|
func NewExistsUserV3WriteModel(resourceOwner, userID string) *UserV3WriteModel {
|
|
return &UserV3WriteModel{
|
|
WriteModel: eventstore.WriteModel{
|
|
AggregateID: userID,
|
|
ResourceOwner: resourceOwner,
|
|
},
|
|
PhoneWM: false,
|
|
EmailWM: false,
|
|
DataWM: false,
|
|
}
|
|
}
|
|
|
|
func NewUserV3WriteModel(resourceOwner, userID string) *UserV3WriteModel {
|
|
return &UserV3WriteModel{
|
|
WriteModel: eventstore.WriteModel{
|
|
AggregateID: userID,
|
|
ResourceOwner: resourceOwner,
|
|
},
|
|
PhoneWM: true,
|
|
EmailWM: true,
|
|
DataWM: true,
|
|
}
|
|
}
|
|
|
|
func (wm *UserV3WriteModel) Reduce() error {
|
|
for _, event := range wm.Events {
|
|
switch e := event.(type) {
|
|
case *schemauser.CreatedEvent:
|
|
wm.SchemaID = e.SchemaID
|
|
wm.SchemaRevision = e.SchemaRevision
|
|
wm.Data = e.Data
|
|
wm.Locked = false
|
|
|
|
wm.State = domain.UserStateActive
|
|
case *schemauser.UpdatedEvent:
|
|
if e.SchemaID != nil {
|
|
wm.SchemaID = *e.SchemaID
|
|
}
|
|
if e.SchemaRevision != nil {
|
|
wm.SchemaRevision = *e.SchemaRevision
|
|
}
|
|
if len(e.Data) > 0 {
|
|
wm.Data = e.Data
|
|
}
|
|
case *schemauser.DeletedEvent:
|
|
wm.State = domain.UserStateDeleted
|
|
case *schemauser.EmailUpdatedEvent:
|
|
wm.Email = string(e.EmailAddress)
|
|
wm.IsEmailVerified = false
|
|
wm.EmailVerifiedFailedCount = 0
|
|
case *schemauser.EmailCodeAddedEvent:
|
|
wm.IsEmailVerified = false
|
|
wm.EmailVerifiedFailedCount = 0
|
|
case *schemauser.EmailVerifiedEvent:
|
|
wm.IsEmailVerified = true
|
|
wm.EmailVerifiedFailedCount = 0
|
|
case *schemauser.EmailVerificationFailedEvent:
|
|
wm.EmailVerifiedFailedCount += 1
|
|
case *schemauser.PhoneUpdatedEvent:
|
|
wm.Phone = string(e.PhoneNumber)
|
|
wm.IsPhoneVerified = false
|
|
wm.PhoneVerifiedFailedCount = 0
|
|
case *schemauser.PhoneCodeAddedEvent:
|
|
wm.IsPhoneVerified = false
|
|
wm.PhoneVerifiedFailedCount = 0
|
|
case *schemauser.PhoneVerifiedEvent:
|
|
wm.PhoneVerifiedFailedCount = 0
|
|
wm.IsPhoneVerified = true
|
|
case *schemauser.PhoneVerificationFailedEvent:
|
|
wm.PhoneVerifiedFailedCount += 1
|
|
case *schemauser.LockedEvent:
|
|
wm.Locked = true
|
|
case *schemauser.UnlockedEvent:
|
|
wm.Locked = false
|
|
case *schemauser.DeactivatedEvent:
|
|
wm.State = domain.UserStateInactive
|
|
case *schemauser.ActivatedEvent:
|
|
wm.State = domain.UserStateActive
|
|
}
|
|
}
|
|
return wm.WriteModel.Reduce()
|
|
}
|
|
|
|
func (wm *UserV3WriteModel) Query() *eventstore.SearchQueryBuilder {
|
|
builder := eventstore.NewSearchQueryBuilder(eventstore.ColumnsEvent)
|
|
if wm.ResourceOwner != "" {
|
|
builder = builder.ResourceOwner(wm.ResourceOwner)
|
|
}
|
|
eventtypes := []eventstore.EventType{
|
|
schemauser.CreatedType,
|
|
schemauser.DeletedType,
|
|
schemauser.ActivatedType,
|
|
schemauser.DeactivatedType,
|
|
schemauser.LockedType,
|
|
schemauser.UnlockedType,
|
|
}
|
|
if wm.DataWM {
|
|
eventtypes = append(eventtypes,
|
|
schemauser.UpdatedType,
|
|
)
|
|
}
|
|
if wm.EmailWM {
|
|
eventtypes = append(eventtypes,
|
|
schemauser.EmailUpdatedType,
|
|
schemauser.EmailVerifiedType,
|
|
schemauser.EmailCodeAddedType,
|
|
schemauser.EmailVerificationFailedType,
|
|
)
|
|
}
|
|
if wm.PhoneWM {
|
|
eventtypes = append(eventtypes,
|
|
schemauser.PhoneUpdatedType,
|
|
schemauser.PhoneVerifiedType,
|
|
schemauser.PhoneCodeAddedType,
|
|
schemauser.PhoneVerificationFailedType,
|
|
)
|
|
}
|
|
return builder.AddQuery().
|
|
AggregateTypes(schemauser.AggregateType).
|
|
AggregateIDs(wm.AggregateID).
|
|
EventTypes(eventtypes...).Builder()
|
|
}
|
|
|
|
func (wm *UserV3WriteModel) NewUpdatedEvent(
|
|
ctx context.Context,
|
|
agg *eventstore.Aggregate,
|
|
schemaID string,
|
|
schemaRevision uint64,
|
|
data json.RawMessage,
|
|
) *schemauser.UpdatedEvent {
|
|
changes := make([]schemauser.Changes, 0)
|
|
if wm.SchemaID != schemaID {
|
|
changes = append(changes, schemauser.ChangeSchemaID(schemaID))
|
|
}
|
|
if wm.SchemaRevision != schemaRevision {
|
|
changes = append(changes, schemauser.ChangeSchemaRevision(schemaRevision))
|
|
}
|
|
if data != nil && !bytes.Equal(wm.Data, data) {
|
|
changes = append(changes, schemauser.ChangeData(data))
|
|
}
|
|
if len(changes) == 0 {
|
|
return nil
|
|
}
|
|
return schemauser.NewUpdatedEvent(ctx, agg, changes)
|
|
}
|
|
|
|
func UserV3AggregateFromWriteModel(wm *eventstore.WriteModel) *eventstore.Aggregate {
|
|
return &eventstore.Aggregate{
|
|
ID: wm.AggregateID,
|
|
Type: schemauser.AggregateType,
|
|
ResourceOwner: wm.ResourceOwner,
|
|
InstanceID: wm.InstanceID,
|
|
Version: schemauser.AggregateVersion,
|
|
}
|
|
}
|
|
|
|
func (wm *UserV3WriteModel) Exists() bool {
|
|
return wm.State != domain.UserStateDeleted && wm.State != domain.UserStateUnspecified
|
|
}
|