login.go 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. package account
  2. import (
  3. "encoding/base64"
  4. "encoding/hex"
  5. "errors"
  6. "fmt"
  7. "mtp20access/client"
  8. "mtp20access/global"
  9. accountModel "mtp20access/model/account"
  10. "mtp20access/model/account/request"
  11. jwtRequest "mtp20access/model/common/request"
  12. "mtp20access/packet"
  13. "sync"
  14. "mtp20access/utils"
  15. "strconv"
  16. "go.uber.org/zap"
  17. )
  18. var (
  19. mtx sync.RWMutex
  20. curSessionID int = 90000 // 本服务SessionID从90000开始,以避免与旧登录服务重叠
  21. )
  22. // Login 用户登录
  23. func Login(req request.LoginReq, addr string) (loginaccount *accountModel.Loginaccount, token string, expiresAt int64, err error) {
  24. // 分别尝试用LoginID、LoginCode和手机号码进行登录
  25. loginaccount, err = getLoginAccount(req.UserName, req.Password)
  26. if err != nil {
  27. return
  28. }
  29. // 判断用户状态
  30. if loginaccount.LOGINSTATUS == 2 {
  31. err = errors.New("账户已冻结")
  32. return
  33. }
  34. if loginaccount.LOGINSTATUS == 3 {
  35. err = errors.New("账户已注销")
  36. return
  37. }
  38. // 生成Token,并写入Redis
  39. if token, expiresAt, err = buildRedisLoginInfo(*loginaccount, addr, req.ClientType); err != nil {
  40. return
  41. }
  42. return
  43. }
  44. // getLoginAccount 分别尝试用LoginID、LoginCode和手机号码进行登录
  45. func getLoginAccount(userName string, password string) (loginaccount *accountModel.Loginaccount, err error) {
  46. // 密码解密(5.0报文解密)
  47. d, err := base64.StdEncoding.DecodeString(password)
  48. if err != nil {
  49. return
  50. }
  51. d1 := d[4 : len(d)-8] // 解密时要去头尾
  52. p, err := packet.Decrypt(d1, packet.AESKey, true)
  53. if err != nil {
  54. return
  55. }
  56. pwd := string(p)
  57. // 通过LoginID查询
  58. if loginID, _ := strconv.Atoi(userName); loginID != 0 {
  59. loginaccount = &accountModel.Loginaccount{
  60. LOGINID: int64(loginID),
  61. PASSWORD: utils.EncoderSha256(fmt.Sprintf("%s%s", userName, pwd)), // 构建数据库存储的密码
  62. }
  63. if has, _ := global.M2A_DB.Get(loginaccount); has {
  64. return
  65. }
  66. }
  67. // 通过LoginCode查询
  68. loginaccount = &accountModel.Loginaccount{
  69. LOGINCODE: userName,
  70. }
  71. if has, _ := global.M2A_DB.Get(loginaccount); has {
  72. // 构建数据库存储的密码
  73. if loginaccount.PASSWORD == utils.EncoderSha256(fmt.Sprintf("%d%s", loginaccount.LOGINID, pwd)) {
  74. return
  75. }
  76. }
  77. // 通过手机号码查询,需要AES加密
  78. key, _ := hex.DecodeString(utils.AESSecretKey)
  79. if mobileEncrypted, _ := utils.AESEncrypt([]byte(userName), key); mobileEncrypted != nil {
  80. loginaccount = &accountModel.Loginaccount{
  81. MOBILE: string(mobileEncrypted),
  82. }
  83. if has, _ := global.M2A_DB.Get(loginaccount); has {
  84. // 构建数据库存储的密码
  85. if loginaccount.PASSWORD == utils.EncoderSha256(fmt.Sprintf("%d%s", loginaccount.LOGINID, pwd)) {
  86. return
  87. }
  88. }
  89. }
  90. err = errors.New("错误的用户名或密码")
  91. return
  92. }
  93. // newSessionID 获取
  94. func newSessionID() int {
  95. mtx.RLock()
  96. defer mtx.RUnlock()
  97. curSessionID += 1
  98. return curSessionID
  99. }
  100. // buildRedisLoginInfo 生成Token,并写入Redis
  101. func buildRedisLoginInfo(loginaccount accountModel.Loginaccount, addr string, group int) (token string, expiresAt int64, err error) {
  102. // 生成SessionID
  103. sessionID := newSessionID()
  104. // 生成本服务Token
  105. j := &utils.JWT{SigningKey: []byte(global.M2A_CONFIG.JWT.SigningKey)} // 唯一签名
  106. claims := j.CreateClaims(jwtRequest.BaseClaims{
  107. LoginID: int(loginaccount.LOGINID),
  108. Group: group,
  109. SessionID: sessionID,
  110. })
  111. token, err = j.CreateToken(claims)
  112. if err != nil {
  113. global.M2A_LOG.Error("生成本服token失败", zap.Error(err))
  114. return
  115. }
  116. expiresAt = claims.RegisteredClaims.ExpiresAt.Unix()
  117. loginLogin := client.LoginRedis{
  118. LoginID: strconv.Itoa(int(loginaccount.LOGINID)),
  119. UserID: strconv.Itoa(int(loginaccount.USERID)),
  120. SessionID: strconv.Itoa(sessionID),
  121. Token: token,
  122. Group: strconv.Itoa(group),
  123. Addr: addr,
  124. }
  125. loginMap, err := loginLogin.ToMap()
  126. // loginMap := map[string]interface{}{
  127. // "LoginID": strconv.Itoa(int(loginaccount.LOGINID)),
  128. // "UserID": strconv.Itoa(int(loginaccount.USERID)),
  129. // "SessionID": strconv.Itoa(sessionID),
  130. // "Token": token,
  131. // "Group": strconv.Itoa(group),
  132. // "Addr": addr,
  133. // }
  134. if err != nil {
  135. global.M2A_LOG.Error("生成登录信息MAP失败", zap.Error(err))
  136. return
  137. }
  138. if err = j.SetRedisLogin(int(loginaccount.LOGINID), group, loginMap); err != nil {
  139. global.M2A_LOG.Error("Token写入Redis失败", zap.Error(err))
  140. return
  141. }
  142. // 生成旧登录服务Token
  143. // if err = j.SetOriRedisToken(int(loginaccount.LOGINID), group); err != nil {
  144. // // FIXME: 这里有类事务的回滚问题
  145. // global.M2A_LOG.Error("生成旧登录服务Token失败", zap.Error(err))
  146. // return
  147. // }
  148. // 记录用户信息
  149. mtx.Lock()
  150. defer mtx.Unlock()
  151. if client.Clients == nil {
  152. client.Clients = make(map[int]*client.Client, 0)
  153. }
  154. delete(client.Clients, claims.SessionID)
  155. client.Clients[claims.SessionID] = &client.Client{LoginRedis: loginLogin}
  156. return
  157. }
  158. // RestoreLoginWithToken 通过Token检验恢复登录状态失败
  159. func RestoreLoginWithToken(loginID int, group int, token string) (err error) {
  160. // 从Redis获取登录信息
  161. j := utils.NewJWT()
  162. loginMap, err := j.GetRedisLogin(loginID, group)
  163. if err != nil {
  164. global.M2A_LOG.Error("Token检验恢复登录状态失败", zap.Error(err))
  165. return
  166. }
  167. loginId := loginMap["loginId"]
  168. userId := loginMap["userId"]
  169. sessionId := loginMap["sessionId"]
  170. addr := loginMap["addr"]
  171. loginLogin := client.LoginRedis{
  172. LoginID: loginId,
  173. UserID: userId,
  174. SessionID: sessionId,
  175. Token: token,
  176. Group: strconv.Itoa(group),
  177. Addr: addr,
  178. }
  179. // 记录用户信息
  180. mtx.Lock()
  181. defer mtx.Unlock()
  182. if client.Clients == nil {
  183. client.Clients = make(map[int]*client.Client, 0)
  184. }
  185. s, err := strconv.Atoi(sessionId)
  186. if err != nil {
  187. global.M2A_LOG.Error("Token检验恢复登录状态失败", zap.Error(err))
  188. return
  189. }
  190. delete(client.Clients, s)
  191. client.Clients[s] = &client.Client{LoginRedis: loginLogin}
  192. return
  193. }