2023-01-06 14:34:53 +08:00
|
|
|
package user_external_login
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-01-09 16:54:20 +08:00
|
|
|
"fmt"
|
2023-01-06 14:34:53 +08:00
|
|
|
"time"
|
|
|
|
|
2023-01-06 17:22:09 +08:00
|
|
|
"github.com/answerdev/answer/internal/base/reason"
|
2023-01-06 14:34:53 +08:00
|
|
|
"github.com/answerdev/answer/internal/entity"
|
|
|
|
"github.com/answerdev/answer/internal/schema"
|
2023-01-09 16:54:20 +08:00
|
|
|
"github.com/answerdev/answer/internal/service/activity"
|
|
|
|
"github.com/answerdev/answer/internal/service/export"
|
|
|
|
"github.com/answerdev/answer/internal/service/siteinfo_common"
|
2023-01-06 14:34:53 +08:00
|
|
|
usercommon "github.com/answerdev/answer/internal/service/user_common"
|
2023-01-09 16:54:20 +08:00
|
|
|
"github.com/answerdev/answer/pkg/random"
|
2023-01-06 17:22:09 +08:00
|
|
|
"github.com/answerdev/answer/pkg/token"
|
2023-01-09 16:54:20 +08:00
|
|
|
"github.com/google/uuid"
|
2023-01-06 17:22:09 +08:00
|
|
|
"github.com/segmentfault/pacman/errors"
|
2023-01-09 16:54:20 +08:00
|
|
|
"github.com/segmentfault/pacman/log"
|
2023-01-06 14:34:53 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
type UserExternalLoginRepo interface {
|
|
|
|
AddUserExternalLogin(ctx context.Context, user *entity.UserExternalLogin) (err error)
|
|
|
|
UpdateInfo(ctx context.Context, userInfo *entity.UserExternalLogin) (err error)
|
|
|
|
GetByExternalID(ctx context.Context, externalID string) (userInfo *entity.UserExternalLogin, exist bool, err error)
|
2023-01-13 12:50:20 +08:00
|
|
|
GetUserExternalLoginList(ctx context.Context, userID string) (
|
|
|
|
resp []*entity.UserExternalLogin, err error)
|
|
|
|
DeleteUserExternalLogin(ctx context.Context, userID, externalID string) (err error)
|
2023-01-09 16:54:20 +08:00
|
|
|
SetCacheUserExternalLoginInfo(ctx context.Context, key string, info *schema.ExternalLoginUserInfoCache) (err error)
|
|
|
|
GetCacheUserExternalLoginInfo(ctx context.Context, key string) (info *schema.ExternalLoginUserInfoCache, err error)
|
2023-01-06 14:34:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// UserExternalLoginService user external login service
|
|
|
|
type UserExternalLoginService struct {
|
|
|
|
userRepo usercommon.UserRepo
|
|
|
|
userExternalLoginRepo UserExternalLoginRepo
|
|
|
|
userCommonService *usercommon.UserCommon
|
2023-01-09 16:54:20 +08:00
|
|
|
emailService *export.EmailService
|
|
|
|
siteInfoCommonService *siteinfo_common.SiteInfoCommonService
|
|
|
|
userActivity activity.UserActiveActivityRepo
|
2023-01-06 14:34:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewUserExternalLoginService new user external login service
|
|
|
|
func NewUserExternalLoginService(
|
|
|
|
userRepo usercommon.UserRepo,
|
|
|
|
userCommonService *usercommon.UserCommon,
|
2023-01-06 14:48:41 +08:00
|
|
|
userExternalLoginRepo UserExternalLoginRepo,
|
2023-01-09 16:54:20 +08:00
|
|
|
emailService *export.EmailService,
|
|
|
|
siteInfoCommonService *siteinfo_common.SiteInfoCommonService,
|
|
|
|
userActivity activity.UserActiveActivityRepo,
|
2023-01-06 14:34:53 +08:00
|
|
|
) *UserExternalLoginService {
|
|
|
|
return &UserExternalLoginService{
|
2023-01-06 14:48:41 +08:00
|
|
|
userRepo: userRepo,
|
|
|
|
userCommonService: userCommonService,
|
|
|
|
userExternalLoginRepo: userExternalLoginRepo,
|
2023-01-09 16:54:20 +08:00
|
|
|
emailService: emailService,
|
|
|
|
siteInfoCommonService: siteInfoCommonService,
|
|
|
|
userActivity: userActivity,
|
2023-01-06 14:34:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExternalLogin if user is already a member logged in
|
|
|
|
func (us *UserExternalLoginService) ExternalLogin(
|
2023-01-09 16:54:20 +08:00
|
|
|
ctx context.Context, externalUserInfo *schema.ExternalLoginUserInfoCache) (
|
2023-01-06 14:34:53 +08:00
|
|
|
resp *schema.UserExternalLoginResp, err error) {
|
2023-01-09 16:54:20 +08:00
|
|
|
oldExternalLoginUserInfo, exist, err := us.userExternalLoginRepo.GetByExternalID(ctx, externalUserInfo.ExternalID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if exist {
|
|
|
|
oldUserInfo, exist, err := us.userRepo.GetByUserID(ctx, oldExternalLoginUserInfo.UserID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if exist {
|
|
|
|
accessToken, _, err := us.userCommonService.CacheLoginUserInfo(
|
|
|
|
ctx, oldUserInfo.ID, oldUserInfo.MailStatus, oldUserInfo.Status)
|
|
|
|
return &schema.UserExternalLoginResp{AccessToken: accessToken}, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-06 14:34:53 +08:00
|
|
|
// cache external user info, waiting for user enter email address.
|
|
|
|
if len(externalUserInfo.Email) == 0 {
|
2023-01-06 17:22:09 +08:00
|
|
|
bindingKey := token.GenerateToken()
|
|
|
|
err = us.userExternalLoginRepo.SetCacheUserExternalLoginInfo(ctx, bindingKey, externalUserInfo)
|
2023-01-06 14:34:53 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-01-06 17:22:09 +08:00
|
|
|
return &schema.UserExternalLoginResp{BindingKey: bindingKey}, nil
|
2023-01-06 14:34:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
oldUserInfo, exist, err := us.userRepo.GetByEmail(ctx, externalUserInfo.Email)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-01-06 14:48:41 +08:00
|
|
|
if !exist {
|
2023-01-09 16:54:20 +08:00
|
|
|
oldUserInfo, err = us.registerNewUser(ctx, externalUserInfo)
|
2023-01-06 14:48:41 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-01-06 14:34:53 +08:00
|
|
|
}
|
2023-01-09 16:54:20 +08:00
|
|
|
err = us.bindOldUser(ctx, externalUserInfo, oldUserInfo)
|
2023-01-06 14:34:53 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
accessToken, _, err := us.userCommonService.CacheLoginUserInfo(
|
|
|
|
ctx, oldUserInfo.ID, oldUserInfo.MailStatus, oldUserInfo.Status)
|
|
|
|
return &schema.UserExternalLoginResp{AccessToken: accessToken}, err
|
|
|
|
}
|
|
|
|
|
2023-01-09 16:54:20 +08:00
|
|
|
func (us *UserExternalLoginService) registerNewUser(ctx context.Context,
|
|
|
|
externalUserInfo *schema.ExternalLoginUserInfoCache) (userInfo *entity.User, err error) {
|
2023-01-06 14:34:53 +08:00
|
|
|
userInfo = &entity.User{}
|
|
|
|
userInfo.EMail = externalUserInfo.Email
|
|
|
|
userInfo.DisplayName = externalUserInfo.Name
|
|
|
|
userInfo.Username, err = us.userCommonService.MakeUsername(ctx, externalUserInfo.Name)
|
|
|
|
if err != nil {
|
2023-01-09 16:54:20 +08:00
|
|
|
userInfo.Username = random.Username()
|
2023-01-06 14:34:53 +08:00
|
|
|
}
|
|
|
|
userInfo.MailStatus = entity.EmailStatusToBeVerified
|
|
|
|
userInfo.Status = entity.UserStatusAvailable
|
|
|
|
userInfo.LastLoginDate = time.Now()
|
|
|
|
err = us.userRepo.AddUser(ctx, userInfo)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return userInfo, nil
|
|
|
|
}
|
|
|
|
|
2023-01-09 16:54:20 +08:00
|
|
|
func (us *UserExternalLoginService) bindOldUser(ctx context.Context,
|
|
|
|
externalUserInfo *schema.ExternalLoginUserInfoCache, oldUserInfo *entity.User) (err error) {
|
2023-01-06 14:34:53 +08:00
|
|
|
oldExternalUserInfo, exist, err := us.userExternalLoginRepo.GetByExternalID(ctx, externalUserInfo.ExternalID)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if exist {
|
|
|
|
oldExternalUserInfo.MetaInfo = externalUserInfo.MetaInfo
|
|
|
|
oldExternalUserInfo.UserID = oldUserInfo.ID
|
|
|
|
err = us.userExternalLoginRepo.UpdateInfo(ctx, oldExternalUserInfo)
|
|
|
|
} else {
|
|
|
|
newExternalUserInfo := &entity.UserExternalLogin{
|
|
|
|
UserID: oldUserInfo.ID,
|
2023-01-06 17:22:09 +08:00
|
|
|
Provider: externalUserInfo.Provider,
|
2023-01-06 14:34:53 +08:00
|
|
|
ExternalID: externalUserInfo.ExternalID,
|
|
|
|
MetaInfo: externalUserInfo.MetaInfo,
|
|
|
|
}
|
|
|
|
err = us.userExternalLoginRepo.AddUserExternalLogin(ctx, newExternalUserInfo)
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
2023-01-06 17:22:09 +08:00
|
|
|
|
2023-01-09 16:54:20 +08:00
|
|
|
// ExternalLoginBindingUserSendEmail Send an email for third-party account login for binding user
|
2023-01-06 17:22:09 +08:00
|
|
|
func (us *UserExternalLoginService) ExternalLoginBindingUserSendEmail(
|
|
|
|
ctx context.Context, req *schema.ExternalLoginBindingUserSendEmailReq) (
|
|
|
|
resp *schema.ExternalLoginBindingUserSendEmailResp, err error) {
|
2023-01-09 16:54:20 +08:00
|
|
|
siteGeneral, err := us.siteInfoCommonService.GetSiteGeneral(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-01-06 17:22:09 +08:00
|
|
|
resp = &schema.ExternalLoginBindingUserSendEmailResp{}
|
|
|
|
externalLoginInfo, err := us.userExternalLoginRepo.GetCacheUserExternalLoginInfo(ctx, req.BindingKey)
|
|
|
|
if err != nil || len(externalLoginInfo.ExternalID) == 0 {
|
|
|
|
return nil, errors.BadRequest(reason.UserNotFound)
|
|
|
|
}
|
2023-01-09 16:54:20 +08:00
|
|
|
if len(externalLoginInfo.Email) > 0 {
|
|
|
|
log.Warnf("the binding email has been sent %s", req.BindingKey)
|
|
|
|
return &schema.ExternalLoginBindingUserSendEmailResp{}, nil
|
|
|
|
}
|
2023-01-06 17:22:09 +08:00
|
|
|
|
2023-01-09 16:54:20 +08:00
|
|
|
userInfo, exist, err := us.userRepo.GetByEmail(ctx, req.Email)
|
2023-01-06 17:22:09 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if exist && !req.Must {
|
|
|
|
resp.EmailExistAndMustBeConfirmed = true
|
|
|
|
return resp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if !exist {
|
|
|
|
externalLoginInfo.Email = req.Email
|
2023-01-09 16:54:20 +08:00
|
|
|
userInfo, err = us.registerNewUser(ctx, externalLoginInfo)
|
2023-01-06 17:22:09 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-01-10 14:24:32 +08:00
|
|
|
resp.AccessToken, _, err = us.userCommonService.CacheLoginUserInfo(
|
|
|
|
ctx, userInfo.ID, userInfo.MailStatus, userInfo.Status)
|
|
|
|
if err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
2023-01-06 17:22:09 +08:00
|
|
|
}
|
2023-01-09 16:54:20 +08:00
|
|
|
err = us.userExternalLoginRepo.SetCacheUserExternalLoginInfo(ctx, req.BindingKey, externalLoginInfo)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-01-06 17:22:09 +08:00
|
|
|
|
2023-01-09 16:54:20 +08:00
|
|
|
// send bind confirmation email
|
|
|
|
data := &schema.EmailCodeContent{
|
|
|
|
SourceType: schema.BindingSourceType,
|
|
|
|
Email: req.Email,
|
|
|
|
UserID: userInfo.ID,
|
|
|
|
BindingKey: req.BindingKey,
|
|
|
|
}
|
|
|
|
code := uuid.NewString()
|
|
|
|
verifyEmailURL := fmt.Sprintf("%s/users/account-activation?code=%s", siteGeneral.SiteUrl, code)
|
|
|
|
title, body, err := us.emailService.RegisterTemplate(ctx, verifyEmailURL)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
go us.emailService.SendAndSaveCode(ctx, userInfo.EMail, title, body, code, data.ToJSONString())
|
2023-01-06 17:22:09 +08:00
|
|
|
return resp, nil
|
|
|
|
}
|
|
|
|
|
2023-01-09 16:54:20 +08:00
|
|
|
// ExternalLoginBindingUser
|
|
|
|
// The user clicks on the email link of the bound account and requests the API to bind the user officially
|
2023-01-06 17:22:09 +08:00
|
|
|
func (us *UserExternalLoginService) ExternalLoginBindingUser(
|
2023-01-09 18:43:52 +08:00
|
|
|
ctx context.Context, bindingKey string, oldUserInfo *entity.User) (err error) {
|
|
|
|
externalLoginInfo, err := us.userExternalLoginRepo.GetCacheUserExternalLoginInfo(ctx, bindingKey)
|
2023-01-09 16:54:20 +08:00
|
|
|
if err != nil || len(externalLoginInfo.ExternalID) == 0 {
|
2023-01-09 18:43:52 +08:00
|
|
|
return errors.BadRequest(reason.UserNotFound)
|
2023-01-09 16:54:20 +08:00
|
|
|
}
|
2023-01-09 18:43:52 +08:00
|
|
|
return us.bindOldUser(ctx, externalLoginInfo, oldUserInfo)
|
2023-01-06 17:22:09 +08:00
|
|
|
}
|
2023-01-13 12:50:20 +08:00
|
|
|
|
|
|
|
// GetExternalLoginUserInfoList get external login user info list
|
|
|
|
func (us *UserExternalLoginService) GetExternalLoginUserInfoList(
|
|
|
|
ctx context.Context, userID string) (resp []*entity.UserExternalLogin, err error) {
|
|
|
|
return us.userExternalLoginRepo.GetUserExternalLoginList(ctx, userID)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExternalLoginUnbinding external login unbinding
|
|
|
|
func (us *UserExternalLoginService) ExternalLoginUnbinding(
|
|
|
|
ctx context.Context, req *schema.ExternalLoginUnbindingReq) (err error) {
|
|
|
|
return us.userExternalLoginRepo.DeleteUserExternalLogin(ctx, req.UserID, req.ExternalID)
|
|
|
|
}
|