auth_controller.go 9.5 KB

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