wxlogin.go 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. package youngee_talent_service
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "fmt"
  6. "github.com/gogf/gf/encoding/gjson"
  7. "net/http"
  8. "youngmini_server/app/dao"
  9. "youngmini_server/app/model"
  10. "youngmini_server/app/model/youngee_talent_model"
  11. "youngmini_server/app/utils"
  12. "github.com/gogf/gf/crypto/gmd5"
  13. "github.com/gogf/gf/frame/g"
  14. "github.com/gogf/gf/net/ghttp"
  15. "github.com/gogf/gf/os/gtime"
  16. )
  17. type WxLoginResult struct {
  18. OpenId string `json:"openid"`
  19. SessionKey string `json:"session_key"`
  20. UnionId string `json:"unionid"`
  21. ErrCode int `json:"errcode"`
  22. ErrMsg string `json:"errmsg"`
  23. }
  24. // 定义 Watermark 结构体
  25. type Watermark struct {
  26. Timestamp int `json:"timestamp"`
  27. AppID string `json:"appid"`
  28. }
  29. // 定义 PhoneInfo 结构体
  30. type PhoneInfo struct {
  31. PhoneNumber string `json:"phoneNumber"`
  32. PurePhoneNumber string `json:"purePhoneNumber"`
  33. CountryCode string `json:"countryCode"`
  34. Watermark Watermark `json:"watermark"`
  35. }
  36. // 定义响应结构体 GetPhoneRes
  37. type GetPhoneRes struct {
  38. ErrCode int `json:"errcode"`
  39. ErrMsg string `json:"errmsg"`
  40. PhoneInfo PhoneInfo `json:"phone_info"`
  41. }
  42. const (
  43. urlgetphone = "https://api.weixin.qq.com/wxa/business/getuserphonenumber?access_token=%s"
  44. urlformat = "https://api.weixin.qq.com/sns/jscode2session?appid=%s&secret=%s&js_code=%s&grant_type=authorization_code"
  45. )
  46. type GetPhoneRequestBody struct {
  47. Code string `json:"code"`
  48. }
  49. func WxLogin(r *ghttp.Request) *TalentHttpResult {
  50. l := youngee_talent_model.WxLoginInfo{}
  51. err := r.ParseForm(&l)
  52. if err != nil {
  53. return &TalentHttpResult{Code: -1, Msg: "param error"}
  54. }
  55. appId := g.Config().GetString("miniapp.appid")
  56. secret := g.Config().GetString("miniapp.appsecret")
  57. url := fmt.Sprintf(urlformat, appId, secret, l.Code)
  58. //通过前端uni.login获得了code用于拼接url
  59. resp, err := http.Get(url)
  60. if err != nil {
  61. return &TalentHttpResult{Code: -2, Msg: err.Error()}
  62. }
  63. //http.Get(url)之后要关闭
  64. defer resp.Body.Close()
  65. // 解码微信服务端传来的信息
  66. wxResp := WxLoginResult{}
  67. decoder := json.NewDecoder(resp.Body)
  68. if err = decoder.Decode(&wxResp); err != nil {
  69. return &TalentHttpResult{Code: -3, Msg: "decode json from wx fail"}
  70. }
  71. if wxResp.ErrCode != 0 {
  72. fmt.Printf("错误码:%d, 错误信息:%s", wxResp.ErrCode, wxResp.ErrMsg)
  73. return &TalentHttpResult{Code: -4, Msg: fmt.Sprintf("errCode:%d, errmsg:%s", wxResp.ErrCode, wxResp.ErrMsg)}
  74. }
  75. //为了获得手机号
  76. phoneNum := getPhoneNum(l.GetPhoneCode)
  77. fmt.Println("phoneNum:---->", phoneNum)
  78. // 根据openid查询达人信息
  79. //rec, err := g.DB().Model("youngee_talent_info").One("talent_phone_number", phoneNum)
  80. rec, err := g.DB().Model("youngee_talent_info").Where("talent_phone_number = ?", phoneNum).One()
  81. if err != nil {
  82. return &TalentHttpResult{Code: -5, Msg: err.Error()}
  83. }
  84. // 如果达人被拉黑,则返回
  85. if rec != nil && rec[dao.YoungeeTalentInfo.Columns.InBlacklist].Int() > 0 {
  86. return &TalentHttpResult{Code: -6, Msg: "in black list"}
  87. }
  88. //获得登录后的返回结果对象
  89. res := youngee_talent_model.LoginResultData{}
  90. var newTalentId string
  91. if rec == nil {
  92. // 如果数据库中不存在此达人,则插入新的达人信息
  93. // 首先生成达人唯一id
  94. newTalentId = utils.GetUuid.GetTalentId()
  95. talentInfo := model.YoungeeTalentInfo{
  96. Id: newTalentId,
  97. TalentWxOpenid: wxResp.OpenId,
  98. Avatar: l.Avatar,
  99. TalentWxNickname: l.Nickname,
  100. TalentPhoneNumber: phoneNum, //插入手机号
  101. Canwithdraw: 0,
  102. Income: 0,
  103. Withdrawing: 0,
  104. Withdrawed: 0,
  105. LastLoginDate: gtime.Now(),
  106. CreateDate: gtime.Now(),
  107. }
  108. res.Avatar = l.Avatar
  109. res.Nickname = l.Nickname
  110. res.Phone = phoneNum //接口返回手机号
  111. _, err = g.DB().Model("youngee_talent_info").Data(talentInfo).Insert()
  112. if err != nil {
  113. return &TalentHttpResult{Code: -7, Msg: "get talentId failed"}
  114. }
  115. } else {
  116. // 如果已存在达人,则读取达人id
  117. newTalentId = rec["id"].String()
  118. res.Avatar = rec["avatar"].String()
  119. res.Nickname = rec["talent_wx_nickname"].String()
  120. res.Phone = rec["talent_phone_number"].String()
  121. // 更新达人最近登录时间
  122. _, err = g.DB().Model("youngee_talent_info").Data(
  123. g.Map{
  124. "last_login_date": gtime.Now(),
  125. }).Where("id", newTalentId).Update()
  126. if err != nil {
  127. return &TalentHttpResult{Code: -8, Msg: err.Error()}
  128. }
  129. }
  130. // 用微信的openid和SessionKey的md5做为token。会一直生效
  131. token, err := gmd5.EncryptString(wxResp.OpenId + wxResp.SessionKey)
  132. fmt.Println("token is " + token)
  133. if err != nil {
  134. return &TalentHttpResult{Code: -9, Msg: "generate key failed"}
  135. }
  136. res.Token = token
  137. // 以token为键保存session 存储在服务器中
  138. //自动生成一个 session ID 并通过 Cookie 将其发送给客户端,以便在后续请求中识别该客户端的会话
  139. //自动生成一个 session ID,并通过 `Set-Cookie` 头将其发送给客户端,前端可以通过r.header['Set-Cookie']获取cookie
  140. err = r.Session.Set(token, g.Map{
  141. "talentId": newTalentId,
  142. "wxOpenId": wxResp.OpenId,
  143. "wxSessionKey": wxResp.SessionKey,
  144. })
  145. if err != nil {
  146. return &TalentHttpResult{Code: -10, Msg: "set session failed"}
  147. }
  148. // 生成自己的session_key返回给小程序
  149. return &TalentHttpResult{Code: 0, Msg: "success", Data: res}
  150. }
  151. func getPhoneNum(code string) string {
  152. //获取access_token
  153. accessToken, err := getAndCacheWxAccessToken()
  154. fmt.Println("accesstoken===>", accessToken)
  155. //post获取电话号码
  156. url := fmt.Sprintf(urlgetphone, accessToken)
  157. fmt.Println("url ", url)
  158. getPhoneRequest := GetPhoneRequestBody{
  159. Code: code,
  160. }
  161. jsonBody, err := gjson.Encode(getPhoneRequest)
  162. // 发送 POST 请求
  163. response, err := http.Post(url, "application/json", bytes.NewBuffer(jsonBody))
  164. if err != nil {
  165. fmt.Printf("post getphoneNum failed")
  166. }
  167. defer response.Body.Close()
  168. // 解码微信服务端传来的信息
  169. var getphoneResp = GetPhoneRes{}
  170. decoder := json.NewDecoder(response.Body)
  171. if err = decoder.Decode(&getphoneResp); err != nil {
  172. fmt.Printf("decode json from wx fail")
  173. }
  174. if getphoneResp.ErrCode != 0 {
  175. fmt.Printf("错误码:%d, 错误信息:%s", getphoneResp.ErrCode, getphoneResp.ErrMsg)
  176. fmt.Sprintf("errCode:%d, errmsg:%s", getphoneResp.ErrCode, getphoneResp.ErrMsg)
  177. }
  178. // 打印解析后的结构体内容
  179. fmt.Println("获取号码接口响应====》", getphoneResp.PhoneInfo.PhoneNumber)
  180. return getphoneResp.PhoneInfo.PhoneNumber
  181. }
  182. func IsLogin(r *ghttp.Request) *TalentHttpResult {
  183. return &TalentHttpResult{Code: 0, Msg: "success!"}
  184. }
  185. func AuthCurrentPhone(r *ghttp.Request) *TalentHttpResult {
  186. tid, err := utils.SessionTalentInfo.GetTalentIdFromSession(r)
  187. if err != nil {
  188. return &TalentHttpResult{Code: -1, Msg: "Get talent info failed"}
  189. }
  190. l := AuthLoginReq{}
  191. err = r.ParseForm(&l)
  192. if err != nil {
  193. r.Response.WriteJson("前端数据解析错误")
  194. }
  195. //通过手机号和验证码构造出key。
  196. VcodeKey := fmt.Sprintf("%s%s", "c_user:", l.Phone)
  197. //viewCount, err := g.Redis().DoVar("GET", projectViewKey)
  198. Vcode, err := g.Redis().DoVar("GET", VcodeKey)
  199. rec, err := g.DB().Model("youngee_talent_info").Where("id", tid).One("talent_phone_number", l.Phone)
  200. //验证码正确
  201. if Vcode.String() == l.Vcode && l.Phone == rec["talent_phone_number"].String() {
  202. //根据电话号码,查询达人信息。达人与电话号码一一对应
  203. return &TalentHttpResult{Code: 0, Msg: "验证成功!"}
  204. } else { //验证码不正确
  205. return &TalentHttpResult{Code: -1, Msg: "验证码错误"}
  206. }
  207. }