153 lines
4.2 KiB
Go
153 lines
4.2 KiB
Go
package app
|
||
|
||
import (
|
||
"net/http"
|
||
"time"
|
||
|
||
"bindbox-game/configs"
|
||
"bindbox-game/internal/code"
|
||
"bindbox-game/internal/pkg/core"
|
||
"bindbox-game/internal/pkg/jwtoken"
|
||
"bindbox-game/internal/pkg/validation"
|
||
"bindbox-game/internal/proposal"
|
||
usersvc "bindbox-game/internal/service/user"
|
||
|
||
"go.uber.org/zap"
|
||
)
|
||
|
||
// ============================================
|
||
// 发送短信验证码
|
||
// ============================================
|
||
|
||
type sendSmsCodeRequest struct {
|
||
Mobile string `json:"mobile" binding:"required"`
|
||
}
|
||
|
||
type sendSmsCodeResponse struct {
|
||
Success bool `json:"success"`
|
||
ExpireSeconds int `json:"expire_seconds"`
|
||
}
|
||
|
||
// SendSmsCode 发送短信验证码
|
||
// @Summary 发送短信验证码
|
||
// @Description 发送短信验证码到指定手机号(60秒内不可重复发送,每日最多10次)
|
||
// @Tags APP端.用户
|
||
// @Accept json
|
||
// @Produce json
|
||
// @Param RequestBody body sendSmsCodeRequest true "请求参数"
|
||
// @Success 200 {object} sendSmsCodeResponse
|
||
// @Failure 400 {object} code.Failure
|
||
// @Router /api/app/sms/send-code [post]
|
||
func (h *handler) SendSmsCode() core.HandlerFunc {
|
||
return func(ctx core.Context) {
|
||
req := new(sendSmsCodeRequest)
|
||
if err := ctx.ShouldBindJSON(req); err != nil {
|
||
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ParamBindError, validation.Error(err)))
|
||
return
|
||
}
|
||
|
||
if err := h.user.SendSmsCode(ctx.RequestContext(), req.Mobile); err != nil {
|
||
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ParamBindError, err.Error()))
|
||
return
|
||
}
|
||
|
||
ctx.Payload(&sendSmsCodeResponse{
|
||
Success: true,
|
||
ExpireSeconds: 300, // 5分钟有效期
|
||
})
|
||
}
|
||
}
|
||
|
||
// ============================================
|
||
// 短信验证码登录
|
||
// ============================================
|
||
|
||
type smsLoginRequest struct {
|
||
Mobile string `json:"mobile" binding:"required"`
|
||
Code string `json:"code" binding:"required"`
|
||
InviteCode string `json:"invite_code"`
|
||
}
|
||
|
||
type smsLoginResponse struct {
|
||
UserID int64 `json:"user_id"`
|
||
Nickname string `json:"nickname"`
|
||
Avatar string `json:"avatar"`
|
||
InviteCode string `json:"invite_code"`
|
||
Mobile string `json:"mobile"`
|
||
Token string `json:"token"`
|
||
OpenID string `json:"openid"`
|
||
IsNewUser bool `json:"is_new_user"`
|
||
}
|
||
|
||
// SmsLogin 短信验证码登录
|
||
// @Summary 短信验证码登录
|
||
// @Description 使用短信验证码登录或注册(新手机号自动创建账户)
|
||
// @Tags APP端.用户
|
||
// @Accept json
|
||
// @Produce json
|
||
// @Param RequestBody body smsLoginRequest true "请求参数"
|
||
// @Success 200 {object} smsLoginResponse
|
||
// @Failure 400 {object} code.Failure
|
||
// @Router /api/app/sms/login [post]
|
||
func (h *handler) SmsLogin() core.HandlerFunc {
|
||
return func(ctx core.Context) {
|
||
req := new(smsLoginRequest)
|
||
rsp := new(smsLoginResponse)
|
||
|
||
if err := ctx.ShouldBindJSON(req); err != nil {
|
||
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ParamBindError, validation.Error(err)))
|
||
return
|
||
}
|
||
|
||
in := usersvc.SmsLoginInput{
|
||
Mobile: req.Mobile,
|
||
Code: req.Code,
|
||
InviteCode: req.InviteCode,
|
||
}
|
||
|
||
out, err := h.user.LoginByCode(ctx.RequestContext(), in)
|
||
if err != nil {
|
||
ctx.AbortWithError(core.Error(http.StatusBadRequest, code.ParamBindError, err.Error()))
|
||
return
|
||
}
|
||
|
||
u := out.User
|
||
rsp.UserID = u.ID
|
||
rsp.Nickname = u.Nickname
|
||
rsp.Avatar = u.Avatar
|
||
rsp.InviteCode = u.InviteCode
|
||
rsp.Mobile = u.Mobile
|
||
rsp.OpenID = u.Openid
|
||
rsp.IsNewUser = out.IsNewUser
|
||
|
||
// 触发邀请奖励逻辑
|
||
if out.IsNewUser && out.InviterID > 0 {
|
||
if err := h.task.OnInviteSuccess(ctx.RequestContext(), out.InviterID, u.ID); err != nil {
|
||
h.logger.Error("触发邀请任务失败", zap.Error(err), zap.Int64("inviter_id", out.InviterID), zap.Int64("invitee_id", u.ID))
|
||
}
|
||
}
|
||
|
||
h.logger.Info("短信登录返回数据",
|
||
zap.Int64("user_id", u.ID),
|
||
zap.String("mobile", u.Mobile),
|
||
zap.String("openid", u.Openid),
|
||
zap.Bool("is_new", out.IsNewUser),
|
||
)
|
||
|
||
// 签发JWT Token
|
||
sessionUserInfo := proposal.SessionUserInfo{
|
||
Id: int32(u.ID),
|
||
UserName: u.Nickname,
|
||
NickName: u.Nickname,
|
||
IsSuper: 0,
|
||
Platform: "APP",
|
||
}
|
||
tokenString, tErr := jwtoken.New(configs.Get().JWT.PatientSecret).Sign(sessionUserInfo, 30*24*time.Hour)
|
||
if tErr == nil {
|
||
rsp.Token = tokenString
|
||
}
|
||
|
||
ctx.Payload(rsp)
|
||
}
|
||
}
|