auth_controller.go 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317
  1. package user
  2. import (
  3. "bufio"
  4. "errors"
  5. "eta/eta_mini_ht_api/common/component/config"
  6. logger "eta/eta_mini_ht_api/common/component/log"
  7. "eta/eta_mini_ht_api/common/contants"
  8. "eta/eta_mini_ht_api/common/exception"
  9. authUtils "eta/eta_mini_ht_api/common/utils/auth"
  10. "eta/eta_mini_ht_api/controllers"
  11. "eta/eta_mini_ht_api/service/auth"
  12. "eta/eta_mini_ht_api/service/user"
  13. "os"
  14. "strings"
  15. )
  16. type AuthController struct {
  17. controllers.BaseController
  18. }
  19. const ChinaAreaCode = "86"
  20. // LoginReq 获取验证码请求
  21. type LoginReq struct {
  22. VerifyCode string `json:"verifyCode"`
  23. Mobile string `json:"mobile"`
  24. AreaCode string `json:"areaCode"`
  25. }
  26. // Logout 小程序退出登录接口
  27. // @Summary 小程序退出登录接口
  28. // @Description 用户小程序退出登录
  29. // @Param mobile body LoginReq true "登录请求体"
  30. // @Success 200 {object} controllers.BaseResponse
  31. // @router /logout [post]
  32. func (a *AuthController) Logout() {
  33. controllers.Wrap(&a.BaseController, func() (result *controllers.WrapData, err error) {
  34. result = a.InitWrapData("退出登录失败")
  35. userInfo := a.Data["user"].(user.User)
  36. err = auth.Logout(userInfo.Id, userInfo.OpenId)
  37. if err != nil {
  38. a.FailedResult("退出登录失败", result)
  39. return
  40. }
  41. return
  42. })
  43. }
  44. // Login 小程序登录接口
  45. // @Summary 小程序用户登录
  46. // @Description 用户通过微信小程序登录
  47. // @Param mobile body LoginReq true "登录请求体"
  48. // @Success 200 {object} controllers.BaseResponse
  49. // @router /login [post]
  50. func (a *AuthController) Login() {
  51. controllers.Wrap(&a.BaseController, func() (result *controllers.WrapData, err error) {
  52. result = a.InitWrapData("登录失败")
  53. loginReq := new(LoginReq)
  54. a.GetPostParams(loginReq)
  55. userInfo := a.Data["user"].(user.User)
  56. if userInfo.Mobile != "" && userInfo.Mobile != loginReq.Mobile {
  57. a.FailedResult("登录失败,已绑定手机号:"+userInfo.Mobile, result)
  58. err = exception.New(exception.IllegalAreaCode)
  59. return
  60. }
  61. mobileUser, err := user.GetUserByMobile(loginReq.Mobile)
  62. var etaErr *exception.EtaError
  63. errors.As(err, &etaErr)
  64. if etaErr.ErrorCode != exception.TemplateUserNotFound {
  65. a.FailedResult("登录失败:"+userInfo.Mobile, result)
  66. err = exception.New(exception.UnknownError)
  67. return
  68. }
  69. if mobileUser.Id > 0 && (mobileUser.Id != userInfo.Id && mobileUser.Mobile == loginReq.Mobile) {
  70. a.FailedResult("登录失败,该手机号已绑定过微信", result)
  71. err = exception.New(exception.IllegalAreaCode)
  72. return
  73. }
  74. if loginReq.AreaCode == "" || !checkValidAreaCode(loginReq.AreaCode) {
  75. a.FailedResult("非法的手机区号", result)
  76. err = exception.New(exception.IllegalAreaCode)
  77. return
  78. }
  79. if loginReq.AreaCode == ChinaAreaCode && !authUtils.IsValidMobile(loginReq.Mobile) {
  80. a.FailedResult("只支持86的区号", result)
  81. err = exception.New(exception.IllegalPhoneNumber)
  82. return
  83. }
  84. err = auth.CheckUser(loginReq.Mobile, loginReq.VerifyCode)
  85. if err != nil {
  86. a.FailedResult("验证码错误", result)
  87. logger.Warn("验证码校验失败:%v", err)
  88. return
  89. }
  90. //用户手机为空需要绑定手机
  91. if userInfo.Mobile == "" {
  92. //注册用户或者登录
  93. err = auth.BindMobile(userInfo.Id, loginReq.Mobile)
  94. if err != nil {
  95. a.FailedResult("登录异常,绑定手机失败", result)
  96. return
  97. }
  98. }
  99. //生成一个正式的登录token,生成登录流水
  100. token, err := auth.Login(userInfo.Id, userInfo.OpenId, loginReq.Mobile, loginReq.VerifyCode)
  101. if err != nil {
  102. a.FailedResult("登录异常,生成accessToken失败", result)
  103. return
  104. }
  105. a.SuccessResult("登录成功", &LoginResp{
  106. Token: token,
  107. }, result)
  108. return
  109. })
  110. }
  111. type LoginResp struct {
  112. Token string
  113. }
  114. // SmsCodeReq 获取验证码请求
  115. type SmsCodeReq struct {
  116. Mobile string `json:"mobile"`
  117. AreaCode string `json:"areaCode"`
  118. }
  119. // SMSCode 小程序手机验证码接口
  120. // @Summary 获取手机验证码
  121. // @Param mobile body SmsCodeReq true "小程序手机验证码接口"
  122. // @Success 200 {object} controllers.BaseResponse
  123. // @Description 用户发送手机验证码
  124. // @router /sendCode [post]
  125. func (a *AuthController) SMSCode() {
  126. controllers.Wrap(&a.BaseController, func() (result *controllers.WrapData, err error) {
  127. result = a.InitWrapData("发送短信失败")
  128. mobile := new(SmsCodeReq)
  129. a.GetPostParams(mobile)
  130. if mobile.AreaCode == "" || !checkValidAreaCode(mobile.AreaCode) {
  131. a.FailedResult("发送短信失败", result)
  132. err = exception.New(exception.IllegalAreaCode)
  133. return
  134. }
  135. phoneNum := mobile.Mobile
  136. if mobile.AreaCode == ChinaAreaCode && !authUtils.IsValidMobile(phoneNum) {
  137. return result, exception.New(exception.IllegalPhoneNumber)
  138. }
  139. //发送短息
  140. err = auth.SendSMSCode(phoneNum)
  141. if err != nil {
  142. logger.Warn("发送短信失败:%v", err)
  143. return result, err
  144. }
  145. if err != nil {
  146. return result, err
  147. }
  148. result = &controllers.WrapData{Msg: "验证码发送成功"}
  149. return result, nil
  150. })
  151. }
  152. type RefreshTokenRes struct {
  153. Token string `json:"token"`
  154. Status string `json:"status"`
  155. IsBindMobile int `json:"isBindMobile"`
  156. TokenType string `json:"tokenType"`
  157. }
  158. // RefreshToken 更新token
  159. // @Summary 更新token
  160. // @Success 200 {object} controllers.BaseResponse
  161. // @Description 更新token
  162. // @router /refreshToken [get]
  163. func (a *AuthController) RefreshToken(code string) {
  164. controllers.Wrap(&a.BaseController, func() (result *controllers.WrapData, err error) {
  165. result = a.InitWrapData("刷新token失败")
  166. if code == "" {
  167. logger.Error("code不能为空")
  168. return result, exception.New(exception.WeChatCodeEmpty)
  169. }
  170. //刷新token
  171. token, isBindMobile, status, tokenType, err := auth.RefreshToken(code)
  172. if err != nil {
  173. logger.Error("刷新token失败:%v", err)
  174. a.FailedResult("刷新token失败", result)
  175. return
  176. }
  177. a.SuccessResult("刷新token成功", RefreshTokenRes{
  178. Token: token,
  179. IsBindMobile: isBindMobile,
  180. TokenType: tokenType,
  181. Status: status,
  182. }, result)
  183. return
  184. })
  185. }
  186. // AreaCodes 小程序手机验证码接口
  187. // @Summary 获取手机验证码
  188. // @Param mobile body SmsCodeReq true "小程序手机验证码接口"
  189. // @Success 200 {object} controllers.BaseResponse
  190. // @Description 用户发送手机验证码
  191. // @router /areaCodes [get]
  192. func (a *AuthController) AreaCodes() {
  193. controllers.Wrap(&a.BaseController, func() (result *controllers.WrapData, err error) {
  194. result = a.InitWrapData("获取区号失败")
  195. list, err := auth.GetAreaCodes()
  196. if err != nil {
  197. a.FailedResult("获取区号失败", result)
  198. }
  199. a.SuccessResult("获取区号成功", list, result)
  200. return
  201. })
  202. }
  203. type WXAppidResp struct {
  204. AppId string
  205. }
  206. // WXAppid 获取APPID
  207. // @Summary 获取APPID
  208. // @Success 200 {object} controllers.BaseResponse
  209. // @Description 获取APPID
  210. // @router /wxAppid [get]
  211. func (a *AuthController) WXAppid() {
  212. controllers.Wrap(&a.BaseController, func() (result *controllers.WrapData, err error) {
  213. result = a.InitWrapData("获取AppId失败")
  214. appid := config.GetConfig(contants.WECHAT).(*config.WechatConfig).GetAppid()
  215. if err != nil {
  216. a.FailedResult("获取AppId失败", result)
  217. }
  218. a.SuccessResult("获取AppId成功", WXAppidResp{
  219. AppId: appid,
  220. }, result)
  221. return
  222. })
  223. }
  224. type FileResp struct {
  225. Content string
  226. }
  227. // Notice 获取注册须知
  228. // @Summary 获取注册须知
  229. // @Success 200 {object} controllers.BaseResponse
  230. // @Description 获取注册须知
  231. // @router /notice [get]
  232. func (a *AuthController) Notice() {
  233. controllers.Wrap(&a.BaseController, func() (result *controllers.WrapData, err error) {
  234. result = a.InitWrapData("获取注册须知失败")
  235. fileConfig := config.GetConfig(contants.FILE).(*config.FileConfig)
  236. //publicKey := config.GetConfig(contants.FILE).(*config.FileConfig).GetPublicKey()
  237. noticeFile := fileConfig.GetNotice()
  238. content, err := readTextConfig(noticeFile)
  239. if err != nil {
  240. err = exception.New(exception.GetNoticeFileError)
  241. a.FailedResult("获取注册须知失败", result)
  242. }
  243. a.SuccessResult("获取注册须知成功", FileResp{
  244. Content: content,
  245. }, result)
  246. return
  247. })
  248. }
  249. // Disclaimer 获取免责声明
  250. // @Summary 获取免责声明
  251. // @Success 200 {object} controllers.BaseResponse
  252. // @Description 获取免责声明
  253. // @router /disclaimer [get]
  254. func (a *AuthController) Disclaimer() {
  255. controllers.Wrap(&a.BaseController, func() (result *controllers.WrapData, err error) {
  256. result = a.InitWrapData("获取免责声明失败")
  257. fileConfig := config.GetConfig(contants.FILE).(*config.FileConfig)
  258. //publicKey := config.GetConfig(contants.FILE).(*config.FileConfig).GetPublicKey()
  259. disclaimerFile := fileConfig.GetDisclaimer()
  260. content, err := readTextConfig(disclaimerFile)
  261. if err != nil {
  262. a.FailedResult("获取免责声明失败", result)
  263. }
  264. a.SuccessResult("获取免责声明成功", FileResp{
  265. Content: content,
  266. }, result)
  267. return
  268. })
  269. }
  270. // 读取纯文本文件
  271. func readTextConfig(filename string) (content string, err error) {
  272. file, err := os.Open(filename)
  273. if err != nil {
  274. return "", err
  275. }
  276. defer file.Close()
  277. scanner := bufio.NewScanner(file)
  278. scanner.Split(bufio.ScanLines)
  279. var lines []string
  280. for scanner.Scan() {
  281. line := scanner.Text()
  282. lines = append(lines, line)
  283. }
  284. err = scanner.Err()
  285. return strings.Join(lines, "\n"), nil
  286. }
  287. func checkValidAreaCode(areaCode string) bool {
  288. list := auth.GetValidAreaCodes()
  289. if areaCode == "" || len(list) == 0 {
  290. return false
  291. }
  292. for _, code := range list {
  293. if areaCode == code {
  294. return true
  295. }
  296. }
  297. return false
  298. }