package api import ( "net/http" "github.com/supabase/auth/internal/api/sms_provider" mail "github.com/supabase/auth/internal/mailer" "github.com/supabase/auth/internal/models" "github.com/supabase/auth/internal/storage" ) // ResendConfirmationParams holds the parameters for a resend request type ResendConfirmationParams struct { Type string `json:"type"` Email string `json:"email"` Phone string `json:"phone"` } func (p *ResendConfirmationParams) Validate(a *API) error { config := a.config switch p.Type { case mail.SignupVerification, mail.EmailChangeVerification, smsVerification, phoneChangeVerification: break default: // type does not match one of the above return badRequestError(ErrorCodeValidationFailed, "Missing one of these types: signup, email_change, sms, phone_change") } if p.Email == "" && p.Type == mail.SignupVerification { return badRequestError(ErrorCodeValidationFailed, "Type provided requires an email address") } if p.Phone == "" && p.Type == smsVerification { return badRequestError(ErrorCodeValidationFailed, "Type provided requires a phone number") } var err error if p.Email != "" && p.Phone != "" { return badRequestError(ErrorCodeValidationFailed, "Only an email address or phone number should be provided.") } else if p.Email != "" { if !config.External.Email.Enabled { return badRequestError(ErrorCodeEmailProviderDisabled, "Email logins are disabled") } p.Email, err = a.validateEmail(p.Email) if err != nil { return err } } else if p.Phone != "" { if !config.External.Phone.Enabled { return badRequestError(ErrorCodePhoneProviderDisabled, "Phone logins are disabled") } p.Phone, err = validatePhone(p.Phone) if err != nil { return err } } else { // both email and phone are empty return badRequestError(ErrorCodeValidationFailed, "Missing email address or phone number") } return nil } // Recover sends a recovery email func (a *API) Resend(w http.ResponseWriter, r *http.Request) error { ctx := r.Context() db := a.db.WithContext(ctx) params := &ResendConfirmationParams{} if err := retrieveRequestParams(r, params); err != nil { return err } if err := params.Validate(a); err != nil { return err } var user *models.User var err error aud := a.requestAud(ctx, r) if params.Email != "" { user, err = models.FindUserByEmailAndAudience(db, params.Email, aud) } else if params.Phone != "" { user, err = models.FindUserByPhoneAndAudience(db, params.Phone, aud) } if err != nil { if models.IsNotFoundError(err) { return sendJSON(w, http.StatusOK, map[string]string{}) } return internalServerError("Unable to process request").WithInternalError(err) } switch params.Type { case mail.SignupVerification: if user.IsConfirmed() { // if the user's email is confirmed already, we don't need to send a confirmation email again return sendJSON(w, http.StatusOK, map[string]string{}) } case smsVerification: if user.IsPhoneConfirmed() { // if the user's phone is confirmed already, we don't need to send a confirmation sms again return sendJSON(w, http.StatusOK, map[string]string{}) } case mail.EmailChangeVerification: // do not resend if user doesn't have a new email address if user.EmailChange == "" { return sendJSON(w, http.StatusOK, map[string]string{}) } case phoneChangeVerification: // do not resend if user doesn't have a new phone number if user.PhoneChange == "" { return sendJSON(w, http.StatusOK, map[string]string{}) } } messageID := "" err = db.Transaction(func(tx *storage.Connection) error { switch params.Type { case mail.SignupVerification: if terr := models.NewAuditLogEntry(r, tx, user, models.UserConfirmationRequestedAction, "", nil); terr != nil { return terr } // PKCE not implemented yet return a.sendConfirmation(r, tx, user, models.ImplicitFlow) case smsVerification: if terr := models.NewAuditLogEntry(r, tx, user, models.UserRecoveryRequestedAction, "", nil); terr != nil { return terr } mID, terr := a.sendPhoneConfirmation(r, tx, user, params.Phone, phoneConfirmationOtp, sms_provider.SMSProvider) if terr != nil { return terr } messageID = mID case mail.EmailChangeVerification: return a.sendEmailChange(r, tx, user, user.EmailChange, models.ImplicitFlow) case phoneChangeVerification: mID, terr := a.sendPhoneConfirmation(r, tx, user, user.PhoneChange, phoneChangeVerification, sms_provider.SMSProvider) if terr != nil { return terr } messageID = mID } return nil }) if err != nil { return err } ret := map[string]any{} if messageID != "" { ret["message_id"] = messageID } return sendJSON(w, http.StatusOK, ret) }