2023-03-29 00:09:06 +02:00
|
|
|
package handlers
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-08-24 11:41:52 +02:00
|
|
|
"strings"
|
2023-03-29 00:09:06 +02:00
|
|
|
"time"
|
|
|
|
|
2023-10-10 15:20:53 +02:00
|
|
|
http_util "github.com/zitadel/zitadel/internal/api/http"
|
2023-08-24 11:41:52 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/api/ui/login"
|
2023-03-29 00:09:06 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/crypto"
|
|
|
|
"github.com/zitadel/zitadel/internal/domain"
|
|
|
|
"github.com/zitadel/zitadel/internal/eventstore"
|
2023-10-19 12:19:10 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/eventstore/handler/v2"
|
2023-03-29 00:09:06 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/notification/types"
|
2023-08-24 11:41:52 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/query"
|
|
|
|
"github.com/zitadel/zitadel/internal/repository/session"
|
2023-03-29 00:09:06 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/repository/user"
|
2023-12-08 16:30:55 +02:00
|
|
|
"github.com/zitadel/zitadel/internal/zerrors"
|
2023-03-29 00:09:06 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
UserNotificationsProjectionTable = "projections.notifications"
|
|
|
|
)
|
|
|
|
|
|
|
|
type userNotifier struct {
|
2023-10-10 15:20:53 +02:00
|
|
|
commands Commands
|
2023-03-29 00:09:06 +02:00
|
|
|
queries *NotificationQueries
|
2023-10-10 15:20:53 +02:00
|
|
|
channels types.ChannelChains
|
2023-08-24 11:41:52 +02:00
|
|
|
otpEmailTmpl string
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewUserNotifier(
|
|
|
|
ctx context.Context,
|
2023-10-19 12:19:10 +02:00
|
|
|
config handler.Config,
|
2023-10-10 15:20:53 +02:00
|
|
|
commands Commands,
|
2023-03-29 00:09:06 +02:00
|
|
|
queries *NotificationQueries,
|
2023-10-10 15:20:53 +02:00
|
|
|
channels types.ChannelChains,
|
2023-08-24 11:41:52 +02:00
|
|
|
otpEmailTmpl string,
|
2023-10-19 12:19:10 +02:00
|
|
|
) *handler.Handler {
|
|
|
|
return handler.NewHandler(ctx, &config, &userNotifier{
|
|
|
|
commands: commands,
|
|
|
|
queries: queries,
|
|
|
|
otpEmailTmpl: otpEmailTmpl,
|
|
|
|
channels: channels,
|
|
|
|
})
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
func (u *userNotifier) Name() string {
|
|
|
|
return UserNotificationsProjectionTable
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) Reducers() []handler.AggregateReducer {
|
2023-03-29 00:09:06 +02:00
|
|
|
return []handler.AggregateReducer{
|
|
|
|
{
|
|
|
|
Aggregate: user.AggregateType,
|
2023-10-19 12:19:10 +02:00
|
|
|
EventReducers: []handler.EventReducer{
|
2023-03-29 00:09:06 +02:00
|
|
|
{
|
|
|
|
Event: user.UserV1InitialCodeAddedType,
|
|
|
|
Reduce: u.reduceInitCodeAdded,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: user.HumanInitialCodeAddedType,
|
|
|
|
Reduce: u.reduceInitCodeAdded,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: user.UserV1EmailCodeAddedType,
|
|
|
|
Reduce: u.reduceEmailCodeAdded,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: user.HumanEmailCodeAddedType,
|
|
|
|
Reduce: u.reduceEmailCodeAdded,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: user.UserV1PasswordCodeAddedType,
|
|
|
|
Reduce: u.reducePasswordCodeAdded,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: user.HumanPasswordCodeAddedType,
|
|
|
|
Reduce: u.reducePasswordCodeAdded,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: user.UserDomainClaimedType,
|
|
|
|
Reduce: u.reduceDomainClaimed,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: user.HumanPasswordlessInitCodeRequestedType,
|
|
|
|
Reduce: u.reducePasswordlessCodeRequested,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: user.UserV1PhoneCodeAddedType,
|
|
|
|
Reduce: u.reducePhoneCodeAdded,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: user.HumanPhoneCodeAddedType,
|
|
|
|
Reduce: u.reducePhoneCodeAdded,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: user.HumanPasswordChangedType,
|
|
|
|
Reduce: u.reducePasswordChanged,
|
|
|
|
},
|
2023-08-15 14:47:05 +02:00
|
|
|
{
|
|
|
|
Event: user.HumanOTPSMSCodeAddedType,
|
|
|
|
Reduce: u.reduceOTPSMSCodeAdded,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: user.HumanOTPEmailCodeAddedType,
|
|
|
|
Reduce: u.reduceOTPEmailCodeAdded,
|
|
|
|
},
|
2023-03-29 00:09:06 +02:00
|
|
|
},
|
|
|
|
},
|
2023-08-24 11:41:52 +02:00
|
|
|
{
|
|
|
|
Aggregate: session.AggregateType,
|
2023-10-19 12:19:10 +02:00
|
|
|
EventReducers: []handler.EventReducer{
|
2023-08-24 11:41:52 +02:00
|
|
|
{
|
|
|
|
Event: session.OTPSMSChallengedType,
|
|
|
|
Reduce: u.reduceSessionOTPSMSChallenged,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Event: session.OTPEmailChallengedType,
|
|
|
|
Reduce: u.reduceSessionOTPEmailChallenged,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) reduceInitCodeAdded(event eventstore.Event) (*handler.Statement, error) {
|
|
|
|
e, ok := event.(*user.HumanInitialCodeAddedEvent)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgumentf(nil, "HANDL-EFe2f", "reduce.wrong.event.type %s", user.HumanInitialCodeAddedType)
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewStatement(event, func(ex handler.Executer, projectionName string) error {
|
|
|
|
ctx := HandlerContext(event.Aggregate())
|
|
|
|
alreadyHandled, err := u.checkIfCodeAlreadyHandledOrExpired(ctx, event, e.Expiry, nil,
|
|
|
|
user.UserV1InitialCodeAddedType, user.UserV1InitialCodeSentType,
|
|
|
|
user.HumanInitialCodeAddedType, user.HumanInitialCodeSentType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if alreadyHandled {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
code, err := crypto.DecryptString(e.Code, u.queries.UserDataCrypto)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
colors, err := u.queries.ActiveLabelPolicyByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-29 00:09:06 +02:00
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
template, err := u.queries.MailTemplateByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-11-21 14:11:38 +02:00
|
|
|
notifyUser, err := u.queries.GetNotifyUserByID(ctx, true, e.Aggregate().ID)
|
2023-10-19 12:19:10 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
translator, err := u.queries.GetTranslatorWithOrgTexts(ctx, notifyUser.ResourceOwner, domain.InitCodeMessageType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, err = u.queries.Origin(ctx, e)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = types.SendEmail(ctx, u.channels, string(template.Template), translator, notifyUser, colors, e).
|
|
|
|
SendUserInitCode(ctx, notifyUser, code)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return u.commands.HumanInitCodeSent(ctx, e.Aggregate().ResourceOwner, e.Aggregate().ID)
|
|
|
|
}), nil
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) reduceEmailCodeAdded(event eventstore.Event) (*handler.Statement, error) {
|
|
|
|
e, ok := event.(*user.HumanEmailCodeAddedEvent)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgumentf(nil, "HANDL-SWf3g", "reduce.wrong.event.type %s", user.HumanEmailCodeAddedType)
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
2023-08-03 06:42:59 +02:00
|
|
|
|
2023-04-25 09:02:29 +02:00
|
|
|
if e.CodeReturned {
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewNoOpStatement(e), nil
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewStatement(event, func(ex handler.Executer, projectionName string) error {
|
|
|
|
ctx := HandlerContext(event.Aggregate())
|
|
|
|
alreadyHandled, err := u.checkIfCodeAlreadyHandledOrExpired(ctx, event, e.Expiry, nil,
|
|
|
|
user.UserV1EmailCodeAddedType, user.UserV1EmailCodeSentType,
|
|
|
|
user.HumanEmailCodeAddedType, user.HumanEmailCodeSentType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if alreadyHandled {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
code, err := crypto.DecryptString(e.Code, u.queries.UserDataCrypto)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
colors, err := u.queries.ActiveLabelPolicyByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-29 00:09:06 +02:00
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
template, err := u.queries.MailTemplateByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-11-21 14:11:38 +02:00
|
|
|
notifyUser, err := u.queries.GetNotifyUserByID(ctx, true, e.Aggregate().ID)
|
2023-10-19 12:19:10 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
translator, err := u.queries.GetTranslatorWithOrgTexts(ctx, notifyUser.ResourceOwner, domain.VerifyEmailMessageType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, err = u.queries.Origin(ctx, e)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = types.SendEmail(ctx, u.channels, string(template.Template), translator, notifyUser, colors, e).
|
|
|
|
SendEmailVerificationCode(ctx, notifyUser, code, e.URLTemplate)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return u.commands.HumanEmailVerificationCodeSent(ctx, e.Aggregate().ResourceOwner, e.Aggregate().ID)
|
|
|
|
}), nil
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) reducePasswordCodeAdded(event eventstore.Event) (*handler.Statement, error) {
|
|
|
|
e, ok := event.(*user.HumanPasswordCodeAddedEvent)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgumentf(nil, "HANDL-Eeg3s", "reduce.wrong.event.type %s", user.HumanPasswordCodeAddedType)
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
2023-06-20 17:34:06 +02:00
|
|
|
if e.CodeReturned {
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewNoOpStatement(e), nil
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewStatement(event, func(ex handler.Executer, projectionName string) error {
|
|
|
|
ctx := HandlerContext(event.Aggregate())
|
|
|
|
alreadyHandled, err := u.checkIfCodeAlreadyHandledOrExpired(ctx, event, e.Expiry, nil,
|
|
|
|
user.UserV1PasswordCodeAddedType, user.UserV1PasswordCodeSentType,
|
|
|
|
user.HumanPasswordCodeAddedType, user.HumanPasswordCodeSentType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if alreadyHandled {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
code, err := crypto.DecryptString(e.Code, u.queries.UserDataCrypto)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
colors, err := u.queries.ActiveLabelPolicyByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-29 00:09:06 +02:00
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
template, err := u.queries.MailTemplateByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-11-21 14:11:38 +02:00
|
|
|
notifyUser, err := u.queries.GetNotifyUserByID(ctx, true, e.Aggregate().ID)
|
2023-10-19 12:19:10 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
translator, err := u.queries.GetTranslatorWithOrgTexts(ctx, notifyUser.ResourceOwner, domain.PasswordResetMessageType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, err = u.queries.Origin(ctx, e)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
notify := types.SendEmail(ctx, u.channels, string(template.Template), translator, notifyUser, colors, e)
|
|
|
|
if e.NotificationType == domain.NotificationTypeSms {
|
|
|
|
notify = types.SendSMSTwilio(ctx, u.channels, translator, notifyUser, colors, e)
|
|
|
|
}
|
|
|
|
err = notify.SendPasswordCode(ctx, notifyUser, code, e.URLTemplate)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return u.commands.PasswordCodeSent(ctx, e.Aggregate().ResourceOwner, e.Aggregate().ID)
|
|
|
|
}), nil
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
2023-08-15 14:47:05 +02:00
|
|
|
func (u *userNotifier) reduceOTPSMSCodeAdded(event eventstore.Event) (*handler.Statement, error) {
|
|
|
|
e, ok := event.(*user.HumanOTPSMSCodeAddedEvent)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgumentf(nil, "HANDL-ASF3g", "reduce.wrong.event.type %s", user.HumanOTPSMSCodeAddedType)
|
2023-08-15 14:47:05 +02:00
|
|
|
}
|
2023-08-24 11:41:52 +02:00
|
|
|
return u.reduceOTPSMS(
|
|
|
|
e,
|
|
|
|
e.Code,
|
|
|
|
e.Expiry,
|
|
|
|
e.Aggregate().ID,
|
|
|
|
e.Aggregate().ResourceOwner,
|
|
|
|
u.commands.HumanOTPSMSCodeSent,
|
|
|
|
user.HumanOTPSMSCodeAddedType,
|
|
|
|
user.HumanOTPSMSCodeSentType,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) reduceSessionOTPSMSChallenged(event eventstore.Event) (*handler.Statement, error) {
|
|
|
|
e, ok := event.(*session.OTPSMSChallengedEvent)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgumentf(nil, "HANDL-Sk32L", "reduce.wrong.event.type %s", session.OTPSMSChallengedType)
|
2023-08-24 11:41:52 +02:00
|
|
|
}
|
|
|
|
if e.CodeReturned {
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewNoOpStatement(e), nil
|
2023-08-24 11:41:52 +02:00
|
|
|
}
|
2023-08-15 14:47:05 +02:00
|
|
|
ctx := HandlerContext(event.Aggregate())
|
2023-08-24 11:41:52 +02:00
|
|
|
s, err := u.queries.SessionByID(ctx, true, e.Aggregate().ID, "")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return u.reduceOTPSMS(
|
|
|
|
e,
|
|
|
|
e.Code,
|
|
|
|
e.Expiry,
|
|
|
|
s.UserFactor.UserID,
|
|
|
|
s.UserFactor.ResourceOwner,
|
|
|
|
u.commands.OTPSMSSent,
|
|
|
|
session.OTPSMSChallengedType,
|
|
|
|
session.OTPSMSSentType,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) reduceOTPSMS(
|
|
|
|
event eventstore.Event,
|
|
|
|
code *crypto.CryptoValue,
|
|
|
|
expiry time.Duration,
|
|
|
|
userID,
|
|
|
|
resourceOwner string,
|
|
|
|
sentCommand func(ctx context.Context, userID string, resourceOwner string) (err error),
|
|
|
|
eventTypes ...eventstore.EventType,
|
|
|
|
) (*handler.Statement, error) {
|
|
|
|
ctx := HandlerContext(event.Aggregate())
|
|
|
|
alreadyHandled, err := u.checkIfCodeAlreadyHandledOrExpired(ctx, event, expiry, nil, eventTypes...)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if alreadyHandled {
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewNoOpStatement(event), nil
|
2023-08-15 14:47:05 +02:00
|
|
|
}
|
2023-08-24 11:41:52 +02:00
|
|
|
plainCode, err := crypto.DecryptString(code, u.queries.UserDataCrypto)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-08-24 11:41:52 +02:00
|
|
|
colors, err := u.queries.ActiveLabelPolicyByOrg(ctx, resourceOwner, false)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-11-21 14:11:38 +02:00
|
|
|
notifyUser, err := u.queries.GetNotifyUserByID(ctx, true, userID)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
translator, err := u.queries.GetTranslatorWithOrgTexts(ctx, notifyUser.ResourceOwner, domain.VerifySMSOTPMessageType)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-10-10 15:20:53 +02:00
|
|
|
ctx, err = u.queries.Origin(ctx, event)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-10-10 15:20:53 +02:00
|
|
|
notify := types.SendSMSTwilio(ctx, u.channels, translator, notifyUser, colors, event)
|
|
|
|
err = notify.SendOTPSMSCode(ctx, plainCode, expiry)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-03-27 18:48:14 +01:00
|
|
|
err = sentCommand(ctx, event.Aggregate().ID, event.Aggregate().ResourceOwner)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewNoOpStatement(event), nil
|
2023-08-15 14:47:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) reduceOTPEmailCodeAdded(event eventstore.Event) (*handler.Statement, error) {
|
|
|
|
e, ok := event.(*user.HumanOTPEmailCodeAddedEvent)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgumentf(nil, "HANDL-JL3hw", "reduce.wrong.event.type %s", user.HumanOTPEmailCodeAddedType)
|
2023-08-15 14:47:05 +02:00
|
|
|
}
|
2023-08-24 11:41:52 +02:00
|
|
|
var authRequestID string
|
|
|
|
if e.AuthRequestInfo != nil {
|
|
|
|
authRequestID = e.AuthRequestInfo.ID
|
|
|
|
}
|
|
|
|
url := func(code, origin string, _ *query.NotifyUser) (string, error) {
|
|
|
|
return login.OTPLink(origin, authRequestID, code, domain.MFATypeOTPEmail), nil
|
|
|
|
}
|
|
|
|
return u.reduceOTPEmail(
|
|
|
|
e,
|
|
|
|
e.Code,
|
|
|
|
e.Expiry,
|
|
|
|
e.Aggregate().ID,
|
|
|
|
e.Aggregate().ResourceOwner,
|
|
|
|
url,
|
|
|
|
u.commands.HumanOTPEmailCodeSent,
|
|
|
|
user.HumanOTPEmailCodeAddedType,
|
|
|
|
user.HumanOTPEmailCodeSentType,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) reduceSessionOTPEmailChallenged(event eventstore.Event) (*handler.Statement, error) {
|
|
|
|
e, ok := event.(*session.OTPEmailChallengedEvent)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgumentf(nil, "HANDL-zbsgt", "reduce.wrong.event.type %s", session.OTPEmailChallengedType)
|
2023-08-24 11:41:52 +02:00
|
|
|
}
|
|
|
|
if e.ReturnCode {
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewNoOpStatement(e), nil
|
2023-08-24 11:41:52 +02:00
|
|
|
}
|
2023-08-15 14:47:05 +02:00
|
|
|
ctx := HandlerContext(event.Aggregate())
|
2023-08-24 11:41:52 +02:00
|
|
|
s, err := u.queries.SessionByID(ctx, true, e.Aggregate().ID, "")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
url := func(code, origin string, user *query.NotifyUser) (string, error) {
|
|
|
|
var buf strings.Builder
|
|
|
|
urlTmpl := origin + u.otpEmailTmpl
|
|
|
|
if e.URLTmpl != "" {
|
|
|
|
urlTmpl = e.URLTmpl
|
|
|
|
}
|
|
|
|
if err := domain.RenderOTPEmailURLTemplate(&buf, urlTmpl, code, user.ID, user.PreferredLoginName, user.DisplayName, user.PreferredLanguage); err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return buf.String(), nil
|
|
|
|
}
|
|
|
|
return u.reduceOTPEmail(
|
|
|
|
e,
|
|
|
|
e.Code,
|
|
|
|
e.Expiry,
|
|
|
|
s.UserFactor.UserID,
|
|
|
|
s.UserFactor.ResourceOwner,
|
|
|
|
url,
|
|
|
|
u.commands.OTPEmailSent,
|
|
|
|
user.HumanOTPEmailCodeAddedType,
|
|
|
|
user.HumanOTPEmailCodeSentType,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) reduceOTPEmail(
|
|
|
|
event eventstore.Event,
|
|
|
|
code *crypto.CryptoValue,
|
|
|
|
expiry time.Duration,
|
|
|
|
userID,
|
|
|
|
resourceOwner string,
|
|
|
|
urlTmpl func(code, origin string, user *query.NotifyUser) (string, error),
|
|
|
|
sentCommand func(ctx context.Context, userID string, resourceOwner string) (err error),
|
|
|
|
eventTypes ...eventstore.EventType,
|
|
|
|
) (*handler.Statement, error) {
|
|
|
|
ctx := HandlerContext(event.Aggregate())
|
|
|
|
alreadyHandled, err := u.checkIfCodeAlreadyHandledOrExpired(ctx, event, expiry, nil, eventTypes...)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if alreadyHandled {
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewNoOpStatement(event), nil
|
2023-08-15 14:47:05 +02:00
|
|
|
}
|
2023-08-24 11:41:52 +02:00
|
|
|
plainCode, err := crypto.DecryptString(code, u.queries.UserDataCrypto)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-08-24 11:41:52 +02:00
|
|
|
colors, err := u.queries.ActiveLabelPolicyByOrg(ctx, resourceOwner, false)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-08-24 11:41:52 +02:00
|
|
|
template, err := u.queries.MailTemplateByOrg(ctx, resourceOwner, false)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-11-21 14:11:38 +02:00
|
|
|
notifyUser, err := u.queries.GetNotifyUserByID(ctx, true, userID)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-08-24 11:41:52 +02:00
|
|
|
translator, err := u.queries.GetTranslatorWithOrgTexts(ctx, resourceOwner, domain.VerifyEmailOTPMessageType)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-10-10 15:20:53 +02:00
|
|
|
ctx, err = u.queries.Origin(ctx, event)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-10-10 15:20:53 +02:00
|
|
|
url, err := urlTmpl(plainCode, http_util.ComposedOrigin(ctx), notifyUser)
|
2023-08-24 11:41:52 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2023-08-15 14:47:05 +02:00
|
|
|
}
|
2023-10-10 15:20:53 +02:00
|
|
|
notify := types.SendEmail(ctx, u.channels, string(template.Template), translator, notifyUser, colors, event)
|
|
|
|
err = notify.SendOTPEmailCode(ctx, url, plainCode, expiry)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-08-24 11:41:52 +02:00
|
|
|
err = sentCommand(ctx, event.Aggregate().ID, event.Aggregate().ResourceOwner)
|
2023-08-15 14:47:05 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewNoOpStatement(event), nil
|
2023-08-15 14:47:05 +02:00
|
|
|
}
|
|
|
|
|
2023-03-29 00:09:06 +02:00
|
|
|
func (u *userNotifier) reduceDomainClaimed(event eventstore.Event) (*handler.Statement, error) {
|
|
|
|
e, ok := event.(*user.DomainClaimedEvent)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgumentf(nil, "HANDL-Drh5w", "reduce.wrong.event.type %s", user.UserDomainClaimedType)
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewStatement(event, func(ex handler.Executer, projectionName string) error {
|
|
|
|
ctx := HandlerContext(event.Aggregate())
|
2024-03-27 18:48:14 +01:00
|
|
|
alreadyHandled, err := u.queries.IsAlreadyHandled(ctx, event, nil,
|
2023-10-19 12:19:10 +02:00
|
|
|
user.UserDomainClaimedType, user.UserDomainClaimedSentType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if alreadyHandled {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
colors, err := u.queries.ActiveLabelPolicyByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-29 00:09:06 +02:00
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
template, err := u.queries.MailTemplateByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-29 00:09:06 +02:00
|
|
|
|
2023-11-21 14:11:38 +02:00
|
|
|
notifyUser, err := u.queries.GetNotifyUserByID(ctx, true, e.Aggregate().ID)
|
2023-10-19 12:19:10 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
translator, err := u.queries.GetTranslatorWithOrgTexts(ctx, notifyUser.ResourceOwner, domain.DomainClaimedMessageType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, err = u.queries.Origin(ctx, e)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = types.SendEmail(ctx, u.channels, string(template.Template), translator, notifyUser, colors, e).
|
|
|
|
SendDomainClaimed(ctx, notifyUser, e.UserName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return u.commands.UserDomainClaimedSent(ctx, e.Aggregate().ResourceOwner, e.Aggregate().ID)
|
|
|
|
}), nil
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) reducePasswordlessCodeRequested(event eventstore.Event) (*handler.Statement, error) {
|
|
|
|
e, ok := event.(*user.HumanPasswordlessInitCodeRequestedEvent)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgumentf(nil, "HANDL-EDtjd", "reduce.wrong.event.type %s", user.HumanPasswordlessInitCodeAddedType)
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
2023-05-24 13:22:00 +03:00
|
|
|
if e.CodeReturned {
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewNoOpStatement(e), nil
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewStatement(event, func(ex handler.Executer, projectionName string) error {
|
|
|
|
ctx := HandlerContext(event.Aggregate())
|
|
|
|
alreadyHandled, err := u.checkIfCodeAlreadyHandledOrExpired(ctx, event, e.Expiry, map[string]interface{}{"id": e.ID}, user.HumanPasswordlessInitCodeSentType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if alreadyHandled {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
code, err := crypto.DecryptString(e.Code, u.queries.UserDataCrypto)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
colors, err := u.queries.ActiveLabelPolicyByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-03-29 00:09:06 +02:00
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
template, err := u.queries.MailTemplateByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-11-21 14:11:38 +02:00
|
|
|
notifyUser, err := u.queries.GetNotifyUserByID(ctx, true, e.Aggregate().ID)
|
2023-10-19 12:19:10 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
translator, err := u.queries.GetTranslatorWithOrgTexts(ctx, notifyUser.ResourceOwner, domain.PasswordlessRegistrationMessageType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, err = u.queries.Origin(ctx, e)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = types.SendEmail(ctx, u.channels, string(template.Template), translator, notifyUser, colors, e).
|
|
|
|
SendPasswordlessRegistrationLink(ctx, notifyUser, code, e.ID, e.URLTemplate)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return u.commands.HumanPasswordlessInitCodeSent(ctx, e.Aggregate().ID, e.Aggregate().ResourceOwner, e.ID)
|
|
|
|
}), nil
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) reducePasswordChanged(event eventstore.Event) (*handler.Statement, error) {
|
|
|
|
e, ok := event.(*user.HumanPasswordChangedEvent)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgumentf(nil, "HANDL-Yko2z8", "reduce.wrong.event.type %s", user.HumanPasswordChangedType)
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewStatement(event, func(ex handler.Executer, projectionName string) error {
|
|
|
|
ctx := HandlerContext(event.Aggregate())
|
2024-03-27 18:48:14 +01:00
|
|
|
alreadyHandled, err := u.queries.IsAlreadyHandled(ctx, event, nil, user.HumanPasswordChangeSentType)
|
2023-10-19 12:19:10 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if alreadyHandled {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
notificationPolicy, err := u.queries.NotificationPolicyByOrg(ctx, true, e.Aggregate().ResourceOwner, false)
|
2023-12-08 16:30:55 +02:00
|
|
|
if zerrors.IsNotFound(err) {
|
2023-10-19 12:19:10 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !notificationPolicy.PasswordChange {
|
|
|
|
return nil
|
|
|
|
}
|
2023-03-29 00:09:06 +02:00
|
|
|
|
|
|
|
colors, err := u.queries.ActiveLabelPolicyByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
2023-10-19 12:19:10 +02:00
|
|
|
return err
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template, err := u.queries.MailTemplateByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
2023-10-19 12:19:10 +02:00
|
|
|
return err
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
2023-11-21 14:11:38 +02:00
|
|
|
notifyUser, err := u.queries.GetNotifyUserByID(ctx, true, e.Aggregate().ID)
|
2023-03-29 00:09:06 +02:00
|
|
|
if err != nil {
|
2023-10-19 12:19:10 +02:00
|
|
|
return err
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
translator, err := u.queries.GetTranslatorWithOrgTexts(ctx, notifyUser.ResourceOwner, domain.PasswordChangeMessageType)
|
|
|
|
if err != nil {
|
2023-10-19 12:19:10 +02:00
|
|
|
return err
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
2023-10-10 15:20:53 +02:00
|
|
|
ctx, err = u.queries.Origin(ctx, e)
|
2023-03-29 00:09:06 +02:00
|
|
|
if err != nil {
|
2023-10-19 12:19:10 +02:00
|
|
|
return err
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
2023-10-10 15:20:53 +02:00
|
|
|
err = types.SendEmail(ctx, u.channels, string(template.Template), translator, notifyUser, colors, e).
|
|
|
|
SendPasswordChange(ctx, notifyUser)
|
2023-03-29 00:09:06 +02:00
|
|
|
if err != nil {
|
2023-10-19 12:19:10 +02:00
|
|
|
return err
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
2023-10-19 12:19:10 +02:00
|
|
|
return u.commands.PasswordChangeSent(ctx, e.Aggregate().ResourceOwner, e.Aggregate().ID)
|
|
|
|
}), nil
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) reducePhoneCodeAdded(event eventstore.Event) (*handler.Statement, error) {
|
|
|
|
e, ok := event.(*user.HumanPhoneCodeAddedEvent)
|
|
|
|
if !ok {
|
2023-12-08 16:30:55 +02:00
|
|
|
return nil, zerrors.ThrowInvalidArgumentf(nil, "HANDL-He83g", "reduce.wrong.event.type %s", user.HumanPhoneCodeAddedType)
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
2023-08-03 06:42:59 +02:00
|
|
|
if e.CodeReturned {
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewNoOpStatement(e), nil
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
2023-10-19 12:19:10 +02:00
|
|
|
return handler.NewStatement(event, func(ex handler.Executer, projectionName string) error {
|
|
|
|
ctx := HandlerContext(event.Aggregate())
|
|
|
|
alreadyHandled, err := u.checkIfCodeAlreadyHandledOrExpired(ctx, event, e.Expiry, nil,
|
|
|
|
user.UserV1PhoneCodeAddedType, user.UserV1PhoneCodeSentType,
|
|
|
|
user.HumanPhoneCodeAddedType, user.HumanPhoneCodeSentType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if alreadyHandled {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
code, err := crypto.DecryptString(e.Code, u.queries.UserDataCrypto)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
colors, err := u.queries.ActiveLabelPolicyByOrg(ctx, e.Aggregate().ResourceOwner, false)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-11-21 14:11:38 +02:00
|
|
|
notifyUser, err := u.queries.GetNotifyUserByID(ctx, true, e.Aggregate().ID)
|
2023-10-19 12:19:10 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
translator, err := u.queries.GetTranslatorWithOrgTexts(ctx, notifyUser.ResourceOwner, domain.VerifyPhoneMessageType)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, err = u.queries.Origin(ctx, e)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = types.SendSMSTwilio(ctx, u.channels, translator, notifyUser, colors, e).
|
|
|
|
SendPhoneVerificationCode(ctx, code)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return u.commands.HumanPhoneVerificationCodeSent(ctx, e.Aggregate().ResourceOwner, e.Aggregate().ID)
|
|
|
|
}), nil
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (u *userNotifier) checkIfCodeAlreadyHandledOrExpired(ctx context.Context, event eventstore.Event, expiry time.Duration, data map[string]interface{}, eventTypes ...eventstore.EventType) (bool, error) {
|
2023-10-19 12:19:10 +02:00
|
|
|
if event.CreatedAt().Add(expiry).Before(time.Now().UTC()) {
|
2023-03-29 00:09:06 +02:00
|
|
|
return true, nil
|
|
|
|
}
|
2024-03-27 18:48:14 +01:00
|
|
|
return u.queries.IsAlreadyHandled(ctx, event, data, eventTypes...)
|
2023-03-29 00:09:06 +02:00
|
|
|
}
|