2023-02-15 02:52:11 +01:00
package command
import (
2023-09-22 11:37:16 +02:00
"errors"
"fmt"
"slices"
"time"
zitadel_errors "github.com/zitadel/zitadel/internal/errors"
2023-02-15 02:52:11 +01:00
"github.com/zitadel/zitadel/internal/eventstore"
2023-09-22 11:37:16 +02:00
"github.com/zitadel/zitadel/internal/id"
2023-02-15 02:52:11 +01:00
"github.com/zitadel/zitadel/internal/repository/quota"
)
type quotaWriteModel struct {
eventstore . WriteModel
2023-09-22 11:37:16 +02:00
rollingAggregateID string
unit quota . Unit
from time . Time
resetInterval time . Duration
amount uint64
limit bool
notifications [ ] * quota . SetEventNotification
2023-02-15 02:52:11 +01:00
}
// newQuotaWriteModel aggregateId is filled by reducing unit matching events
func newQuotaWriteModel ( instanceId , resourceOwner string , unit quota . Unit ) * quotaWriteModel {
return & quotaWriteModel {
WriteModel : eventstore . WriteModel {
InstanceID : instanceId ,
ResourceOwner : resourceOwner ,
} ,
unit : unit ,
}
}
func ( wm * quotaWriteModel ) Query ( ) * eventstore . SearchQueryBuilder {
query := eventstore . NewSearchQueryBuilder ( eventstore . ColumnsEvent ) .
ResourceOwner ( wm . ResourceOwner ) .
InstanceID ( wm . InstanceID ) .
2023-10-19 12:19:10 +02:00
AddQuery ( ) .
2023-02-15 02:52:11 +01:00
AggregateTypes ( quota . AggregateType ) .
EventTypes (
quota . AddedEventType ,
2023-09-22 11:37:16 +02:00
quota . SetEventType ,
2023-02-15 02:52:11 +01:00
quota . RemovedEventType ,
) . EventData ( map [ string ] interface { } { "unit" : wm . unit } )
return query . Builder ( )
}
func ( wm * quotaWriteModel ) Reduce ( ) error {
for _ , event := range wm . Events {
2023-10-19 12:19:10 +02:00
wm . ChangeDate = event . CreatedAt ( )
2023-02-15 02:52:11 +01:00
switch e := event . ( type ) {
2023-09-22 11:37:16 +02:00
case * quota . SetEvent :
wm . rollingAggregateID = e . Aggregate ( ) . ID
if e . Amount != nil {
wm . amount = * e . Amount
}
if e . From != nil {
wm . from = * e . From
}
if e . Limit != nil {
wm . limit = * e . Limit
}
if e . ResetInterval != nil {
wm . resetInterval = * e . ResetInterval
}
if e . Notifications != nil {
wm . notifications = * e . Notifications
}
2023-02-15 02:52:11 +01:00
case * quota . RemovedEvent :
2023-09-22 11:37:16 +02:00
wm . rollingAggregateID = ""
2023-02-15 02:52:11 +01:00
}
}
2023-09-22 11:37:16 +02:00
if err := wm . WriteModel . Reduce ( ) ; err != nil {
return err
}
// wm.WriteModel.Reduce() sets the aggregateID to the first event's aggregateID, but we need the last one
wm . AggregateID = wm . rollingAggregateID
return nil
}
// NewChanges returns all changes that need to be applied to the aggregate.
// If createNew is true, all quota properties are set.
func ( wm * quotaWriteModel ) NewChanges (
idGenerator id . Generator ,
createNew bool ,
amount uint64 ,
from time . Time ,
resetInterval time . Duration ,
limit bool ,
notifications ... * QuotaNotification ,
) ( changes [ ] quota . QuotaChange , err error ) {
setEventNotifications , err := QuotaNotifications ( notifications ) . newSetEventNotifications ( idGenerator )
if err != nil {
return nil , err
}
// we sort the input notifications already, so we can return early if they have duplicates
err = sortSetEventNotifications ( setEventNotifications )
if err != nil {
return nil , err
}
if createNew {
return [ ] quota . QuotaChange {
quota . ChangeAmount ( amount ) ,
quota . ChangeFrom ( from ) ,
quota . ChangeResetInterval ( resetInterval ) ,
quota . ChangeLimit ( limit ) ,
quota . ChangeNotifications ( setEventNotifications ) ,
} , nil
}
changes = make ( [ ] quota . QuotaChange , 0 , 5 )
if wm . amount != amount {
changes = append ( changes , quota . ChangeAmount ( amount ) )
}
if wm . from != from {
changes = append ( changes , quota . ChangeFrom ( from ) )
}
if wm . resetInterval != resetInterval {
changes = append ( changes , quota . ChangeResetInterval ( resetInterval ) )
}
if wm . limit != limit {
changes = append ( changes , quota . ChangeLimit ( limit ) )
}
// If the number of notifications differs, we renew the notifications and we can return early
if len ( setEventNotifications ) != len ( wm . notifications ) {
changes = append ( changes , quota . ChangeNotifications ( setEventNotifications ) )
return changes , nil
}
// Now we sort the existing notifications too, so comparing the input properties with the existing ones is easier.
// We ignore the sorting error for the existing notifications, because this is system state, not user input.
// If sorting fails this time, the notifications are listed in the event payload and the projection cleans them up anyway.
_ = sortSetEventNotifications ( wm . notifications )
for i , notification := range setEventNotifications {
if notification . CallURL != wm . notifications [ i ] . CallURL ||
notification . Percent != wm . notifications [ i ] . Percent ||
notification . Repeat != wm . notifications [ i ] . Repeat {
changes = append ( changes , quota . ChangeNotifications ( setEventNotifications ) )
return changes , nil
}
}
return changes , err
}
// newSetEventNotifications returns quota.SetEventNotification elements with generated IDs.
func ( q QuotaNotifications ) newSetEventNotifications ( idGenerator id . Generator ) ( setNotifications [ ] * quota . SetEventNotification , err error ) {
if q == nil {
return make ( [ ] * quota . SetEventNotification , 0 ) , nil
}
notifications := make ( [ ] * quota . SetEventNotification , len ( q ) )
for idx , notification := range q {
notifications [ idx ] = & quota . SetEventNotification {
Percent : notification . Percent ,
Repeat : notification . Repeat ,
CallURL : notification . CallURL ,
}
notifications [ idx ] . ID , err = idGenerator . Next ( )
if err != nil {
return nil , err
}
}
return notifications , nil
}
// sortSetEventNotifications reports an error if there are duplicate notifications or if a pointer is nil
func sortSetEventNotifications ( notifications [ ] * quota . SetEventNotification ) ( err error ) {
slices . SortFunc ( notifications , func ( i , j * quota . SetEventNotification ) int {
if i == nil || j == nil {
err = zitadel_errors . ThrowInternal ( errors . New ( "sorting slices of *quota.SetEventNotification with nil pointers is not supported" ) , "QUOTA-8YXPk" , "Errors.Internal" )
return 0
}
if i . Percent == j . Percent && i . CallURL == j . CallURL && i . Repeat == j . Repeat {
// TODO: translate
err = zitadel_errors . ThrowInternal ( fmt . Errorf ( "%+v" , i ) , "QUOTA-Pty2n" , "Errors.Quota.Notifications.Duplicate" )
return 0
}
if i . Percent < j . Percent ||
i . Percent == j . Percent && i . CallURL < j . CallURL ||
2023-09-29 11:26:14 +02:00
i . Percent == j . Percent && i . CallURL == j . CallURL && ! i . Repeat && j . Repeat {
2023-09-22 11:37:16 +02:00
return - 1
}
return + 1
} )
return err
2023-02-15 02:52:11 +01:00
}