Files
open-im-server/internal/api/auth/auth.go
T

234 lines
10 KiB
Go
Raw Normal View History

2021-12-27 18:22:32 +08:00
package apiAuth
import (
2023-01-30 15:28:46 +08:00
api "Open_IM/pkg/api_struct"
2021-12-27 18:22:32 +08:00
"Open_IM/pkg/common/config"
2022-09-09 10:29:30 +08:00
"Open_IM/pkg/common/constant"
2021-12-27 18:22:32 +08:00
"Open_IM/pkg/common/log"
2023-02-09 14:40:49 +08:00
"Open_IM/pkg/common/tokenverify"
2021-12-27 18:22:32 +08:00
rpc "Open_IM/pkg/proto/auth"
2023-02-09 20:36:34 +08:00
open_im_sdk "Open_IM/pkg/proto/sdkws"
2021-12-28 17:45:57 +08:00
"Open_IM/pkg/utils"
2021-12-27 18:22:32 +08:00
"context"
"net/http"
"strings"
2022-08-07 22:37:27 +08:00
"github.com/fatih/structs"
"github.com/gin-gonic/gin"
2021-12-27 18:22:32 +08:00
)
2022-06-29 18:48:17 +08:00
// @Summary 用户注册
// @Description 用户注册
2022-06-29 11:18:19 +08:00
// @Tags 鉴权认证
// @ID UserRegister
// @Accept json
2022-06-29 18:48:17 +08:00
// @Param req body api.UserRegisterReq true "secret为openIM密钥, 详细见服务端config.yaml secret字段 <br> platform为平台ID <br> ex为拓展字段 <br> gender为性别, 0为女, 1为男"
2022-06-29 11:18:19 +08:00
// @Produce json
// @Success 0 {object} api.UserRegisterResp
2022-06-29 18:48:17 +08:00
// @Failure 500 {object} api.Swagger500Resp "errCode为500 一般为服务器内部错误"
// @Failure 400 {object} api.Swagger400Resp "errCode为400 一般为参数输入错误, token未带上等"
2022-06-29 11:18:19 +08:00
// @Router /auth/user_register [post]
2021-12-27 18:22:32 +08:00
func UserRegister(c *gin.Context) {
params := api.UserRegisterReq{}
if err := c.BindJSON(&params); err != nil {
2022-04-20 20:58:51 +08:00
errMsg := " BindJSON failed " + err.Error()
log.NewError("0", errMsg)
c.JSON(http.StatusBadRequest, gin.H{"errCode": 400, "errMsg": errMsg})
2021-12-27 18:22:32 +08:00
return
}
if params.Secret != config.Config.Secret {
2022-04-20 18:39:12 +08:00
errMsg := " params.Secret != config.Config.Secret "
log.NewError(params.OperationID, errMsg, params.Secret, config.Config.Secret)
c.JSON(http.StatusBadRequest, gin.H{"errCode": 401, "errMsg": errMsg})
2021-12-27 18:22:32 +08:00
return
}
2021-12-28 16:36:02 +08:00
req := &rpc.UserRegisterReq{UserInfo: &open_im_sdk.UserInfo{}}
2021-12-28 17:49:52 +08:00
utils.CopyStructFields(req.UserInfo, &params)
2021-12-28 17:48:21 +08:00
//copier.Copy(req.UserInfo, &params)
2021-12-28 16:03:47 +08:00
req.OperationID = params.OperationID
2021-12-28 16:01:57 +08:00
log.NewInfo(req.OperationID, "UserRegister args ", req.String())
2023-02-07 20:24:20 +08:00
etcdConn := rpc.GetDefaultConn(config.Config.Etcd.EtcdSchema, strings.Join(config.Config.Etcd.EtcdAddr, ","), config.Config.RpcRegisterName.OpenImAuthName, req.OperationID)
2022-06-16 14:09:28 +08:00
if etcdConn == nil {
2022-08-17 12:12:54 +08:00
errMsg := req.OperationID + " getcdv3.GetDefaultConn == nil"
2022-06-16 14:09:28 +08:00
log.NewError(req.OperationID, errMsg)
c.JSON(http.StatusInternalServerError, gin.H{"errCode": 500, "errMsg": errMsg})
return
}
2021-12-27 18:22:32 +08:00
client := rpc.NewAuthClient(etcdConn)
reply, err := client.UserRegister(context.Background(), req)
2022-04-01 20:40:58 +08:00
if err != nil {
2022-05-07 11:55:19 +08:00
errMsg := req.OperationID + " " + "UserRegister failed " + err.Error() + req.String()
2022-04-20 20:58:51 +08:00
log.NewError(req.OperationID, errMsg)
c.JSON(http.StatusInternalServerError, gin.H{"errCode": 500, "errMsg": errMsg})
2022-04-01 20:40:58 +08:00
return
}
if reply.CommonResp.ErrCode != 0 {
2022-05-07 11:55:19 +08:00
errMsg := req.OperationID + " " + " UserRegister failed " + reply.CommonResp.ErrMsg + req.String()
2022-04-20 20:58:51 +08:00
log.NewError(req.OperationID, errMsg)
2022-09-09 10:29:30 +08:00
if reply.CommonResp.ErrCode == constant.RegisterLimit {
c.JSON(http.StatusOK, gin.H{"errCode": constant.RegisterLimit, "errMsg": "用户注册被限制"})
} else if reply.CommonResp.ErrCode == constant.InvitationError {
c.JSON(http.StatusOK, gin.H{"errCode": constant.InvitationError, "errMsg": "邀请码错误"})
} else {
c.JSON(http.StatusInternalServerError, gin.H{"errCode": 500, "errMsg": errMsg})
}
2021-12-27 18:22:32 +08:00
return
}
2022-09-02 11:26:25 +08:00
pbDataToken := &rpc.UserTokenReq{Platform: params.Platform, FromUserID: params.UserID, OperationID: params.OperationID}
2021-12-27 18:22:32 +08:00
replyToken, err := client.UserToken(context.Background(), pbDataToken)
if err != nil {
2022-04-20 20:58:51 +08:00
errMsg := req.OperationID + " " + " client.UserToken failed " + err.Error() + pbDataToken.String()
log.NewError(req.OperationID, errMsg)
c.JSON(http.StatusInternalServerError, gin.H{"errCode": 500, "errMsg": errMsg})
2021-12-27 18:22:32 +08:00
return
}
resp := api.UserRegisterResp{CommResp: api.CommResp{ErrCode: replyToken.CommonResp.ErrCode, ErrMsg: replyToken.CommonResp.ErrMsg},
UserToken: api.UserTokenInfo{UserID: req.UserInfo.UserID, Token: replyToken.Token, ExpiredTime: replyToken.ExpiredTime}}
log.NewInfo(req.OperationID, "UserRegister return ", resp)
2021-12-28 18:03:51 +08:00
c.JSON(http.StatusOK, resp)
2021-12-27 18:22:32 +08:00
}
2022-06-29 11:18:19 +08:00
// @Summary 用户登录
// @Description 获取用户的token
// @Tags 鉴权认证
// @ID UserToken
// @Accept json
2022-06-29 18:48:17 +08:00
// @Param req body api.UserTokenReq true "secret为openIM密钥, 详细见服务端config.yaml secret字段 <br> platform为平台ID"
2022-06-29 11:18:19 +08:00
// @Produce json
// @Success 0 {object} api.UserTokenResp
2022-06-29 18:48:17 +08:00
// @Failure 500 {object} api.Swagger500Resp "errCode为500 一般为服务器内部错误"
// @Failure 400 {object} api.Swagger400Resp "errCode为400 一般为参数输入错误, token未带上等"
2022-06-29 11:18:19 +08:00
// @Router /auth/user_token [post]
2021-12-27 18:22:32 +08:00
func UserToken(c *gin.Context) {
params := api.UserTokenReq{}
if err := c.BindJSON(&params); err != nil {
2022-04-20 20:58:51 +08:00
errMsg := " BindJSON failed " + err.Error()
2022-09-12 19:32:24 +08:00
log.NewError(params.OperationID, errMsg)
2022-04-20 20:58:51 +08:00
c.JSON(http.StatusBadRequest, gin.H{"errCode": 400, "errMsg": errMsg})
2021-12-27 18:22:32 +08:00
return
}
if params.Secret != config.Config.Secret {
2022-04-20 20:58:51 +08:00
errMsg := params.OperationID + " params.Secret != config.Config.Secret "
2021-12-27 18:22:32 +08:00
log.NewError(params.OperationID, "params.Secret != config.Config.Secret", params.Secret, config.Config.Secret)
2022-04-20 20:58:51 +08:00
c.JSON(http.StatusBadRequest, gin.H{"errCode": 401, "errMsg": errMsg})
2021-12-27 18:22:32 +08:00
return
}
2022-08-07 22:37:27 +08:00
req := &rpc.UserTokenReq{Platform: params.Platform, FromUserID: params.UserID, OperationID: params.OperationID, LoginIp: params.LoginIp}
2021-12-28 16:01:57 +08:00
log.NewInfo(req.OperationID, "UserToken args ", req.String())
2023-02-07 20:24:20 +08:00
etcdConn := rpc.GetDefaultConn(config.Config.Etcd.EtcdSchema, strings.Join(config.Config.Etcd.EtcdAddr, ","), config.Config.RpcRegisterName.OpenImAuthName, req.OperationID)
2022-06-16 14:09:28 +08:00
if etcdConn == nil {
2022-08-17 12:12:54 +08:00
errMsg := req.OperationID + " getcdv3.GetDefaultConn == nil"
2022-06-16 14:09:28 +08:00
log.NewError(req.OperationID, errMsg)
c.JSON(http.StatusInternalServerError, gin.H{"errCode": 500, "errMsg": errMsg})
return
}
2021-12-27 18:22:32 +08:00
client := rpc.NewAuthClient(etcdConn)
reply, err := client.UserToken(context.Background(), req)
if err != nil {
2022-06-21 16:36:34 +08:00
errMsg := req.OperationID + " UserToken failed " + err.Error() + " req: " + req.String()
2022-04-20 20:58:51 +08:00
log.NewError(req.OperationID, errMsg)
c.JSON(http.StatusInternalServerError, gin.H{"errCode": 500, "errMsg": errMsg})
2021-12-27 18:22:32 +08:00
return
}
resp := api.UserTokenResp{CommResp: api.CommResp{ErrCode: reply.CommonResp.ErrCode, ErrMsg: reply.CommonResp.ErrMsg},
UserToken: api.UserTokenInfo{UserID: req.FromUserID, Token: reply.Token, ExpiredTime: reply.ExpiredTime}}
2022-04-20 18:47:21 +08:00
log.NewInfo(req.OperationID, "UserToken return ", resp)
2021-12-28 18:18:35 +08:00
c.JSON(http.StatusOK, resp)
2021-12-27 18:22:32 +08:00
}
2022-05-10 09:45:55 +08:00
2022-06-29 18:48:17 +08:00
// @Summary 解析当前用户token
// @Description 解析当前用户token(token在请求头中传入)
2022-06-29 11:18:19 +08:00
// @Tags 鉴权认证
// @ID ParseToken
// @Accept json
// @Param token header string true "im token"
2022-06-29 18:48:17 +08:00
// @Param req body api.ParseTokenReq true "secret为openIM密钥, 详细见服务端config.yaml secret字段<br>platform为平台ID"
2022-06-29 11:18:19 +08:00
// @Produce json
2023-02-09 19:40:45 +08:00
// @Success 0 {object} api.ParseTokenResp{Map=api.ExpireTime}
2022-06-29 18:48:17 +08:00
// @Failure 500 {object} api.Swagger500Resp "errCode为500 一般为服务器内部错误"
// @Failure 400 {object} api.Swagger400Resp "errCode为400 一般为参数输入错误, token未带上等"
2022-06-29 11:18:19 +08:00
// @Router /auth/parse_token [post]
2022-05-10 09:45:55 +08:00
func ParseToken(c *gin.Context) {
params := api.ParseTokenReq{}
if err := c.BindJSON(&params); err != nil {
errMsg := " BindJSON failed " + err.Error()
log.NewError("0", errMsg)
2022-07-29 17:49:15 +08:00
c.JSON(http.StatusOK, gin.H{"errCode": 1001, "errMsg": errMsg})
2022-05-10 09:45:55 +08:00
return
}
var ok bool
var errInfo string
var expireTime int64
2023-02-09 14:40:49 +08:00
ok, _, errInfo, expireTime = tokenverify.GetUserIDFromTokenExpireTime(c.Request.Header.Get("token"), params.OperationID)
2022-05-10 09:45:55 +08:00
if !ok {
2022-07-29 17:58:51 +08:00
errMsg := params.OperationID + " " + "GetUserIDFromTokenExpireTime failed " + errInfo
2022-05-10 09:45:55 +08:00
log.NewError(params.OperationID, errMsg)
2022-07-29 17:49:15 +08:00
c.JSON(http.StatusOK, gin.H{"errCode": 1001, "errMsg": errMsg})
2022-05-10 09:45:55 +08:00
return
}
2022-05-10 12:07:09 +08:00
resp := api.ParseTokenResp{CommResp: api.CommResp{ErrCode: 0, ErrMsg: ""}, ExpireTime: api.ExpireTime{ExpireTimeSeconds: uint32(expireTime)}}
2022-05-10 12:20:53 +08:00
resp.Data = structs.Map(&resp.ExpireTime)
2022-05-10 09:45:55 +08:00
log.NewInfo(params.OperationID, "ParseToken return ", resp)
c.JSON(http.StatusOK, resp)
}
2022-06-06 20:39:45 +08:00
2022-06-29 11:18:19 +08:00
// @Summary 强制登出
// @Description 对应的平台强制登出
// @Tags 鉴权认证
// @ID ForceLogout
// @Accept json
// @Param token header string true "im token"
2022-06-29 18:48:17 +08:00
// @Param req body api.ForceLogoutReq true "platform为平台ID <br> fromUserID为要执行强制登出的用户ID"
2022-06-29 11:18:19 +08:00
// @Produce json
// @Success 0 {object} api.ForceLogoutResp
2022-06-29 18:48:17 +08:00
// @Failure 500 {object} api.Swagger500Resp "errCode为500 一般为服务器内部错误"
// @Failure 400 {object} api.Swagger400Resp "errCode为400 一般为参数输入错误, token未带上等"
2022-06-29 11:18:19 +08:00
// @Router /auth/force_logout [post]
2022-06-06 20:39:45 +08:00
func ForceLogout(c *gin.Context) {
params := api.ForceLogoutReq{}
if err := c.BindJSON(&params); err != nil {
errMsg := " BindJSON failed " + err.Error()
log.NewError("0", errMsg)
c.JSON(http.StatusBadRequest, gin.H{"errCode": 400, "errMsg": errMsg})
return
}
2022-06-07 11:42:15 +08:00
req := &rpc.ForceLogoutReq{}
utils.CopyStructFields(req, &params)
2022-06-06 20:39:45 +08:00
var ok bool
var errInfo string
2023-02-09 14:40:49 +08:00
ok, req.OpUserID, errInfo = tokenverify.GetUserIDFromToken(c.Request.Header.Get("token"), req.OperationID)
2022-06-06 20:39:45 +08:00
if !ok {
2022-06-07 11:42:15 +08:00
errMsg := req.OperationID + " " + "GetUserIDFromToken failed " + errInfo + " token:" + c.Request.Header.Get("token")
log.NewError(req.OperationID, errMsg)
2022-06-06 20:39:45 +08:00
c.JSON(http.StatusInternalServerError, gin.H{"errCode": 500, "errMsg": errMsg})
return
}
2022-06-07 11:42:15 +08:00
log.NewInfo(req.OperationID, "ForceLogout args ", req.String())
2023-02-07 20:24:20 +08:00
etcdConn := rpc.GetDefaultConn(config.Config.Etcd.EtcdSchema, strings.Join(config.Config.Etcd.EtcdAddr, ","), config.Config.RpcRegisterName.OpenImAuthName, req.OperationID)
2022-06-16 14:09:28 +08:00
if etcdConn == nil {
2022-08-17 12:12:54 +08:00
errMsg := req.OperationID + " getcdv3.GetDefaultConn == nil"
2022-06-16 14:09:28 +08:00
log.NewError(req.OperationID, errMsg)
c.JSON(http.StatusInternalServerError, gin.H{"errCode": 500, "errMsg": errMsg})
return
}
2022-06-07 11:42:15 +08:00
client := rpc.NewAuthClient(etcdConn)
reply, err := client.ForceLogout(context.Background(), req)
if err != nil {
errMsg := req.OperationID + " UserToken failed " + err.Error() + req.String()
log.NewError(req.OperationID, errMsg)
c.JSON(http.StatusInternalServerError, gin.H{"errCode": 500, "errMsg": errMsg})
return
}
2022-06-07 16:38:42 +08:00
resp := api.ForceLogoutResp{CommResp: api.CommResp{ErrCode: reply.CommonResp.ErrCode, ErrMsg: reply.CommonResp.ErrMsg}}
log.NewInfo(params.OperationID, utils.GetSelfFuncName(), " return ", resp)
c.JSON(http.StatusOK, resp)
2022-06-06 20:39:45 +08:00
}