Files
sendico/api/edge/bff/internal/server/invitationimp/accept.go
2026-02-28 00:39:20 +01:00

137 lines
5.1 KiB
Go

package invitationimp
import (
"context"
"encoding/json"
"errors"
"net/http"
"github.com/tech/sendico/pkg/api/http/response"
"github.com/tech/sendico/pkg/merrors"
an "github.com/tech/sendico/pkg/messaging/notifications/account"
"github.com/tech/sendico/pkg/model"
"github.com/tech/sendico/pkg/mutil/mzap"
"github.com/tech/sendico/server/interface/api/srequest"
"go.mongodb.org/mongo-driver/v2/bson"
"go.uber.org/zap"
)
func (a *InvitationAPI) doAccept(ctx context.Context, invitationRef bson.ObjectID, accData *model.AccountData) error {
inv, err := a.getPendingInvitation(ctx, invitationRef)
if err != nil {
return err
}
org, err := a.getOrganization(ctx, inv.OrganizationRef, inv.Content.Email)
if err != nil {
return err
}
if _, err := a.fetchOrCreateAccount(ctx, org, inv, accData); err != nil {
return err
}
if err := a.db.Accept(ctx, invitationRef); err != nil {
a.Logger.Warn("Failed to accept invitation", zap.Error(err), mzap.ObjRef("invitation_ref", invitationRef))
return err
}
return nil
}
func (a *InvitationAPI) sendWelcomeEmail(account *model.Account, token string) error {
if err := a.producer.SendMessage(an.AccountCreated(a.Name(), *account.GetID(), token)); err != nil {
a.Logger.Warn("Failed to send account creation notification", zap.Error(err))
return err
}
return nil
}
func (a *InvitationAPI) getPendingInvitation(ctx context.Context, invitationRef bson.ObjectID) (*model.Invitation, error) {
a.Logger.Debug("Fetching invitation", mzap.ObjRef("invitation_ref", invitationRef))
var inv model.Invitation
if err := a.db.Unprotected().Get(ctx, invitationRef, &inv); err != nil {
a.Logger.Warn("Failed to fetch invitation", zap.Error(err), mzap.ObjRef("invitation_ref", invitationRef))
return nil, err
}
if inv.Status != model.InvitationCreated {
a.Logger.Warn("Invitation is not pending", mzap.StorableRef(&inv))
return nil, merrors.InvalidArgument("Invitation is not pending")
}
return &inv, nil
}
func (a *InvitationAPI) getOrganization(ctx context.Context, orgRef bson.ObjectID, email string) (*model.Organization, error) {
a.Logger.Debug("Fetching organization", mzap.ObjRef("organization_ref", orgRef), zap.String("email", email))
var org model.Organization
if err := a.odb.Unprotected().Get(ctx, orgRef, &org); err != nil {
a.Logger.Warn("Failed to fetch organization when processing invitation", zap.Error(err),
mzap.ObjRef("organization_ref", orgRef), zap.String("email", email))
return nil, err
}
return &org, nil
}
func (a *InvitationAPI) fetchOrCreateAccount(ctx context.Context, org *model.Organization, inv *model.Invitation, accData *model.AccountData) (*model.Account, error) {
account, err := a.adb.GetByEmail(ctx, inv.Content.Email)
if errors.Is(err, merrors.ErrNoData) {
a.Logger.Debug("Account is not registered, creating", zap.String("email", inv.Content.Email))
if accData == nil {
a.Logger.Warn("Account data missing for unregistered invitation acceptance",
zap.String("email", inv.Content.Email), mzap.StorableRef(inv))
return nil, merrors.InvalidArgument("No account data provided for invitation acceptance")
}
account = accData.ToAccount()
if err := a.accService.ValidateAccount(account); err != nil {
a.Logger.Info("Account validation failed", zap.Error(err), zap.String("email", inv.Content.Email))
return nil, err
}
// creates account and joins organization
token, err := a.accService.CreateAccount(ctx, org, account, inv.RoleRef)
if err != nil {
a.Logger.Warn("Failed to create account", zap.Error(err), zap.String("email", inv.Content.Email))
return nil, err
}
// Send welcome email
if err = a.sendWelcomeEmail(account, token); err != nil {
a.Logger.Warn("Failed to send welcome email for new account created via invitation",
zap.Error(err), zap.String("email", inv.Content.Email))
return nil, err
}
return account, nil
} else if err != nil {
a.Logger.Warn("Failed to fetch account by email", zap.Error(err), zap.String("email", inv.Content.Email))
return nil, err
} else {
// If account already exists, then just join organization
if err := a.accService.JoinOrganization(ctx, org, account, inv.RoleRef); err != nil {
a.Logger.Warn("Failed to join organization", zap.Error(err), mzap.StorableRef(account), mzap.StorableRef(org))
return nil, err
}
}
return account, nil
}
func (a *InvitationAPI) accept(r *http.Request) http.HandlerFunc {
invitationRef, err := a.irh.GetRef(r)
if err != nil {
return a.respondBadReference(r, err)
}
var req srequest.AcceptInvitation
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
a.Logger.Warn("Failed to decode request body", zap.Error(err), mzap.ObjRef("invitation_ref", invitationRef))
return response.BadPayload(a.Logger, a.Name(), err)
}
if _, err := a.tf.CreateTransaction().Execute(r.Context(), func(ctx context.Context) (any, error) {
return nil, a.doAccept(ctx, invitationRef, req.Account)
}); err != nil {
a.Logger.Warn("Failed to accept invitation", zap.Error(err), mzap.ObjRef("invitation_ref", invitationRef))
return response.Auto(a.Logger, a.Name(), err)
}
return response.Success(a.Logger)
}