user.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370
  1. package services
  2. import (
  3. "errors"
  4. "hongze/hongze_api/models"
  5. "hongze/hongze_api/utils"
  6. "strconv"
  7. "time"
  8. )
  9. var ERR_NO_USER_RECORD = errors.New("用户关系没有入库")
  10. var ERR_USER_NOT_BIND = errors.New("用户没有绑定")
  11. //通过openid获取用户信息
  12. func GetWxUserItemByOpenId(openid string) (item *models.WxUserItem,err error){
  13. //通过openid获取用户关联信息
  14. userRecord,userRecordErr := models.GetUserRecordByOpenId(openid)
  15. if userRecordErr != nil{
  16. if userRecordErr.Error() == utils.ErrNoRow(){
  17. err = ERR_NO_USER_RECORD
  18. return
  19. }else{
  20. err = userRecordErr
  21. return
  22. }
  23. }
  24. //该openid在系统中没有关联关系
  25. if userRecord == nil {
  26. err = ERR_NO_USER_RECORD
  27. return
  28. }
  29. //该openid没有绑定用户
  30. if userRecord.UserId <= 0{
  31. err = ERR_USER_NOT_BIND
  32. item = new(models.WxUserItem)
  33. //格式化返回用户数据
  34. formatWxUserAndUserRecord(item,userRecord)
  35. return
  36. }
  37. //获取用户信息
  38. item,wxUserErr := models.GetWxUserItemByUserId(userRecord.UserId)
  39. if wxUserErr != nil{
  40. err = wxUserErr
  41. return
  42. }
  43. //格式化返回用户数据
  44. formatWxUserAndUserRecord(item,userRecord)
  45. return
  46. }
  47. //根据用户id和平台id获取用户信息
  48. func GetWxUserItemByUserId(userId,platform int) (wxUserItem *models.WxUserItem,err error){
  49. //获取用户信息
  50. wxUserItem,wxUserErr := models.GetWxUserItemByUserId(userId)
  51. if wxUserErr != nil{
  52. err = wxUserErr
  53. return
  54. }
  55. //格式化返回用户数据
  56. formatWxUser(wxUserItem,platform)
  57. return
  58. }
  59. //根据用户邮箱和平台id获取用户信息
  60. func GetWxUserItemByEmail(email string,platform int) (wxUserItem *models.WxUserItem,err error) {
  61. //获取用户信息
  62. wxUserItem,wxUserErr := models.GetWxUserItemByEmail(email)
  63. if wxUserErr != nil{
  64. err = wxUserErr
  65. return
  66. }
  67. //格式化返回用户数据
  68. formatWxUser(wxUserItem,platform)
  69. return
  70. }
  71. //根据用户手机号和平台id获取用户信息
  72. func GetWxUserItemByMobile(mobile string,platform int) (wxUserItem *models.WxUserItem,err error) {
  73. //获取用户信息
  74. wxUserItem,wxUserErr := models.GetWxUserItemByMobile(mobile)
  75. if wxUserErr != nil{
  76. err = wxUserErr
  77. return
  78. }
  79. //格式化返回用户数据
  80. formatWxUser(wxUserItem,platform)
  81. return
  82. }
  83. //根据用户unionid和平台id获取用户信息
  84. func GetWxUserItemByUnionId(unionId string,platform int) (wxUserItem *models.WxUserItem,err error) {
  85. //获取用户信息
  86. wxUserItem,wxUserErr := models.GetWxUserItemByUnionid(unionId)
  87. if wxUserErr != nil{
  88. err = wxUserErr
  89. return
  90. }
  91. //格式化返回用户数据
  92. formatWxUser(wxUserItem,platform)
  93. return
  94. }
  95. //通过用户 关系表记录 和 用户记录 格式化返回 用户数据
  96. func formatWxUserAndUserRecord(wxUser *models.WxUserItem,userRecord *models.UserRecord) {
  97. wxUser.OpenId = userRecord.OpenId
  98. wxUser.UnionId = userRecord.UnionId
  99. wxUser.NickName = userRecord.NickName
  100. //wxUser.RealName = userRecord.RealName
  101. //wxUser.BindAccount = userRecord.BindAccount
  102. wxUser.Headimgurl = userRecord.Headimgurl
  103. }
  104. //通过用户 用户记录 和 来源平台 格式化返回 用户数据
  105. func formatWxUser(wxUser *models.WxUserItem,platform int) {
  106. //根据用户id和平台id获取用户关系
  107. userRecord,userRecordErr := models.GetUserRecordByUserId(wxUser.UserId,platform)
  108. if userRecordErr != nil{
  109. if userRecordErr.Error() != utils.ErrNoRow(){
  110. return
  111. }
  112. if userRecordErr.Error() == utils.ErrNoRow(){
  113. return
  114. }
  115. }
  116. //该openid在系统中没有关联关系
  117. if userRecord == nil {
  118. return
  119. }
  120. wxUser.OpenId = userRecord.OpenId
  121. wxUser.UnionId = userRecord.UnionId
  122. wxUser.NickName = userRecord.NickName
  123. //wxUser.RealName = userRecord.RealName
  124. //wxUser.BindAccount = userRecord.BindAccount
  125. wxUser.Headimgurl = userRecord.Headimgurl
  126. return
  127. }
  128. //用户绑定
  129. func BindWxUser(openid,mobile,email string) (wxUser *models.WxUserItem,err error) {
  130. if mobile == "" && email == ""{
  131. err = errors.New("手机号或邮箱必填一个")
  132. return
  133. }
  134. var bindAccount string
  135. //根据手机号获取用户信息
  136. if mobile != ""{
  137. tmpWxUser,wxUserErr := models.GetWxUserItemByMobile(mobile)
  138. if wxUserErr != nil && wxUserErr.Error() != utils.ErrNoRow(){
  139. err = wxUserErr
  140. return
  141. }
  142. wxUser = tmpWxUser
  143. bindAccount = mobile
  144. }
  145. //根据邮箱获取用户信息
  146. if wxUser == nil && email != ""{
  147. tmpWxUser,wxUserErr := models.GetWxUserItemByEmail(email)
  148. if wxUserErr != nil && wxUserErr.Error() != utils.ErrNoRow(){
  149. err = wxUserErr
  150. return
  151. }
  152. wxUser = tmpWxUser
  153. bindAccount = email
  154. }
  155. //查询openid的第三方(微信)信息
  156. userRecord,err := models.GetUserRecordByOpenId(openid)
  157. if err != nil{
  158. return
  159. }
  160. var userId int
  161. //如果查询出来的用户是nil,那么需要新增用户
  162. if wxUser == nil{
  163. user := &models.WxUser{
  164. CompanyId:1,
  165. CreatedTime:time.Now(),
  166. FirstLogin:1,
  167. Enabled:1,
  168. RegisterPlatform:1,
  169. RegisterTime:time.Now(),
  170. Mobile: mobile,
  171. Email: email,
  172. }
  173. tmpUserId, addUserErr := models.AddWxUser(user)
  174. if err != nil{
  175. err = addUserErr
  176. return
  177. }
  178. user.UserId = int(tmpUserId)
  179. userId = int(tmpUserId)
  180. wxUser,err = models.GetWxUserItemByUserId(userId)
  181. }else{
  182. userId = wxUser.UserId
  183. }
  184. //如果存在该手机号/邮箱,那么需要校验
  185. if userRecord.UserId > 0 && userRecord.UserId != userId{
  186. err = errors.New("用户已绑定,不允许重复绑定")
  187. return
  188. }
  189. if userRecord.UserId == 0{
  190. err = models.BindUserRecordByOpenid(userId,openid,bindAccount)
  191. if err != nil{
  192. return
  193. }
  194. userRecord.UserId = userId
  195. }
  196. //如果当前该第三方用户信息的昵称为空串的话,那么需要去查询该用户的第一个绑定信息的数据作为来源做数据修复
  197. if userRecord.NickName == ""{
  198. oldUserRecord, err := models.GetUserThirdRecordByUserId(userId)
  199. if err == nil && oldUserRecord != nil{
  200. //如果该用户绑定的第一条数据的头像信息不为空串,那么就去做新数据的修复
  201. if oldUserRecord.NickName != ""{
  202. _ = models.ModifyUserRecordInfo(userRecord.OpenId,oldUserRecord.NickName, oldUserRecord.Headimgurl, oldUserRecord.City, oldUserRecord.Province, oldUserRecord.Country, oldUserRecord.Sex, userId)
  203. }
  204. }
  205. }
  206. //格式化用户数据
  207. formatWxUserAndUserRecord(wxUser,userRecord)
  208. return
  209. }
  210. //微信登录
  211. func WxLogin(wxPlatform int,code string,wxAccessToken *WxAccessToken,wxUserInfo *WxUserInfo)(token string,userId,firstLogin,permission int,err error){
  212. openId := wxAccessToken.Openid
  213. unionId := wxAccessToken.Unionid
  214. if unionId == ""{
  215. unionId = wxUserInfo.Unionid
  216. }
  217. //firstLogin==1,强制绑定手机号或者邮箱
  218. firstLogin = 1
  219. QUERY_WX_USER:
  220. wxUser, wxUserErr := GetWxUserItemByOpenId(openId)
  221. if wxUserErr == ERR_NO_USER_RECORD{ //没有用户openid记录
  222. _,recordErr := AddUserRecord(openId,unionId,wxUserInfo.Nickname,"",wxUserInfo.Province,wxUserInfo.City,wxUserInfo.Country,wxUserInfo.Headimgurl,"",wxPlatform,wxUserInfo.Sex,0)
  223. //如果插入失败,那么直接将错误信息返回
  224. if recordErr != nil{
  225. err = recordErr
  226. return
  227. }
  228. //插入成功后,需要重新查询该用户,并进入下面的逻辑
  229. goto QUERY_WX_USER
  230. }else if wxUserErr == ERR_USER_NOT_BIND{
  231. //没有用户信息
  232. //wxUser.FirstLogin = 1
  233. }else if wxUserErr != nil{
  234. err = wxUserErr
  235. return
  236. }
  237. //如果已经登录注册绑定的情况下
  238. if wxUser != nil && wxUserErr == nil{
  239. //获取用户权限
  240. firstLogin = wxUser.FirstLogin
  241. userId = wxUser.UserId
  242. permission, permissionErr := CheckUserPermission(userId)
  243. if permissionErr != nil {
  244. //记录日志
  245. utils.FileLog.Info("userId:%s,err:%s", strconv.Itoa(userId), err)
  246. }
  247. if wxUserInfo != nil {
  248. go models.ModifyUserRecordInfo(openId,wxUserInfo.Nickname, wxUserInfo.Headimgurl, wxUserInfo.City, wxUserInfo.Province, wxUserInfo.Country, wxUserInfo.Sex, userId)
  249. }
  250. {
  251. codeLog := new(models.WxUserCode)
  252. codeLog.WxCode = code
  253. codeLog.UserId = userId
  254. codeLog.Code = 0
  255. codeLog.FirstLogin = firstLogin
  256. codeLog.Authorization = token
  257. codeLog.UserPermission = permission
  258. codeLog.CreateTime=time.Now()
  259. models.AddWxUserCode(codeLog)
  260. }
  261. if wxUser.Mobile == "" && wxUser.Email == "" {
  262. firstLogin = 1
  263. }
  264. }
  265. //获取登录token
  266. tokenItem, tokenErr := models.GetTokenByOpenId(openId)
  267. if tokenErr != nil && tokenErr.Error() != utils.ErrNoRow() {
  268. err = errors.New("登录失败,获取token失败:" + tokenErr.Error())
  269. return
  270. }
  271. if tokenItem == nil || (tokenErr != nil && tokenErr.Error() == utils.ErrNoRow()) {
  272. timeUnix := time.Now().Unix()
  273. timeUnixStr := strconv.FormatInt(timeUnix, 10)
  274. token = utils.MD5(openId) + utils.MD5(timeUnixStr)
  275. //新增session
  276. {
  277. session := new(models.Session)
  278. session.OpenId = openId
  279. session.UserId = userId
  280. session.CreatedTime = time.Now()
  281. session.LastUpdatedTime = time.Now()
  282. session.ExpireTime = time.Now().AddDate(0, 3, 0)
  283. session.AccessToken = token
  284. sessionErr := models.AddSession(session)
  285. if err != nil {
  286. err = errors.New("登录失败,新增用户session信息失败:" + sessionErr.Error())
  287. return
  288. }
  289. }
  290. } else {
  291. token = tokenItem.AccessToken
  292. }
  293. //新增登录日志
  294. {
  295. loginLog := new(models.WxUserLog)
  296. loginLog.UserId = userId
  297. loginLog.OpenId = openId
  298. loginLog.UnionId = unionId
  299. loginLog.CreateTime = time.Now()
  300. loginLog.Handle = "wechat_login"
  301. loginLog.Remark = token
  302. go models.AddWxUserLog(loginLog)
  303. }
  304. return
  305. }
  306. //添加第三方用户(微信)记录
  307. func AddUserRecord(openId,unionId,nickName,realName,province,city,country,headimgurl,sessionKey string,platform,sex,subscribe int) (userRecord *models.UserRecord,err error) {
  308. find,err := models.GetUserRecordByOpenId(openId)
  309. if err != nil && err.Error() != utils.ErrNoRow(){
  310. return
  311. }
  312. if find != nil{
  313. userRecord = find
  314. return
  315. }
  316. userRecord = &models.UserRecord{
  317. OpenId :openId, //用户open_id
  318. UnionId :unionId,//用户union_id
  319. Subscribe :subscribe,
  320. NickName :nickName, //用户昵称,最大长度:32
  321. RealName :realName, //用户实际名称,最大长度:32
  322. Sex :sex, //普通用户性别,1为男性,2为女性
  323. Province :province,//普通用户个人资料填写的省份,最大长度:30
  324. City :city, //普通用户个人资料填写的城市,最大长度:30
  325. Country :country, //国家,如中国为CN,最大长度:30
  326. Headimgurl :headimgurl, //用户第三方(微信)头像,最大长度:512
  327. CreateTime :time.Now(), //创建时间,关系添加时间、用户授权时间
  328. CreatePlatform :platform, //注册平台,1:日度点评公众号,2:管理后台,3:pc端网站,4:查研观向小程序;默认:1
  329. SessionKey :sessionKey, //微信小程序会话密钥,最大长度:255
  330. }
  331. recordId, err := models.AddUserRecord(userRecord)
  332. if err !=nil{
  333. return
  334. }
  335. userRecord.UserRecordId = int(recordId)
  336. return
  337. }