wechat.go 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327
  1. package services
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "github.com/rdlucklib/rdluck_tools/http"
  7. "hongze/hongze_api/models"
  8. "hongze/hongze_api/services/go_redis"
  9. "hongze/hongze_api/utils"
  10. "strconv"
  11. "strings"
  12. "time"
  13. )
  14. type WxAccessToken struct {
  15. AccessToken string `json:"access_token"`
  16. ExpiresIn int `json:"expires_in"`
  17. RefreshToken string `json:"refresh_token"`
  18. Openid string `json:"openid"`
  19. Unionid string `json:"unionid"`
  20. Scope string `json:"scope"`
  21. Errcode int `json:"errcode"`
  22. Errmsg string `json:"errmsg"`
  23. }
  24. func WxGetUserOpenIdByCode(code string) (item *WxAccessToken, err error) {
  25. if code == "" {
  26. err = errors.New("code is empty")
  27. return nil, err
  28. }
  29. requestUrl := `https://api.weixin.qq.com/sns/oauth2/access_token?appid=%s&secret=%s&code=%s&grant_type=authorization_code`
  30. requestUrl = fmt.Sprintf(requestUrl, utils.WxAppId, utils.WxAppSecret, code)
  31. result, err := http.Get(requestUrl)
  32. if err != nil {
  33. return nil, err
  34. }
  35. utils.FileLog.Info("WxGetUserOpenIdByCode:%s", string(result))
  36. err = json.Unmarshal(result, &item)
  37. return
  38. }
  39. type WxToken struct {
  40. AccessToken string `json:"access_token"`
  41. ExpiresIn int `json:"expires_in"`
  42. Errcode int `json:"errcode"`
  43. Errmsg string `json:"errmsg"`
  44. }
  45. func WxGetToken(wxAppId, wxAppSecret string) (item *WxToken, err error) {
  46. requestUrl := `https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s`
  47. requestUrl = fmt.Sprintf(requestUrl, wxAppId, wxAppSecret)
  48. fmt.Println("requestUrl:", requestUrl)
  49. result, err := http.Get(requestUrl)
  50. if err != nil {
  51. return nil, err
  52. }
  53. err = json.Unmarshal(result, &item)
  54. fmt.Println("WxGetToken start")
  55. fmt.Println(string(result))
  56. fmt.Println("WxGetToken end")
  57. return
  58. }
  59. /*func WxGetAccessToken() (accessToken string, err error) {
  60. wxToken, err := models.GetWxToken()
  61. fmt.Println(err, wxToken)
  62. if err != nil && err.Error() != utils.ErrNoRow() {
  63. return
  64. }
  65. //wx_token 不存在
  66. if wxToken == nil || (err != nil && err.Error() == utils.ErrNoRow()) {
  67. token, err := WxGetToken()
  68. if err != nil {
  69. return accessToken, err
  70. }
  71. if token.Errmsg != "" {
  72. err = errors.New("获取access_token 失败 errcode:" + token.Errmsg + " ;errmsg:" + token.Errmsg)
  73. return "", err
  74. }
  75. expiresIn := time.Now().Add(110 * time.Minute).Unix()
  76. err = models.AddWxToken(token.AccessToken, expiresIn)
  77. if err != nil {
  78. err = errors.New("新增wx_token失败" + err.Error())
  79. return accessToken, err
  80. }
  81. accessToken = token.AccessToken
  82. } else {
  83. if wxToken.ExpiresIn <= time.Now().Unix() {
  84. token, err := WxGetToken()
  85. if err != nil {
  86. return accessToken, err
  87. }
  88. if token.Errmsg != "" {
  89. err = errors.New("获取access_token 失败 errcode:" + token.Errmsg + " ;errmsg:" + token.Errmsg)
  90. return "", err
  91. }
  92. expiresIn := time.Now().Add(110 * time.Minute).Unix()
  93. err = models.UpdateWxToken(token.AccessToken, expiresIn, wxToken.Id)
  94. if err != nil {
  95. err = errors.New("修改wx_token失败" + err.Error())
  96. return accessToken, err
  97. }
  98. accessToken = token.AccessToken
  99. } else {
  100. accessToken = wxToken.AccessToken
  101. }
  102. }
  103. return
  104. }*/
  105. type WxUserInfo struct {
  106. Openid string `json:"openid"`
  107. Nickname string `json:"nickname"`
  108. Sex int `json:"sex"`
  109. Language string `json:"language"`
  110. City string `json:"city"`
  111. Province string `json:"province"`
  112. Country string `json:"country"`
  113. Headimgurl string `json:"headimgurl"`
  114. SubscribeTime int `json:"subscribe_time"`
  115. Unionid string `json:"unionid"`
  116. Remark string `json:"remark"`
  117. Groupid int `json:"groupid"`
  118. SubscribeScene string `json:"subscribe_scene"`
  119. Errcode int `json:"errcode"`
  120. Errmsg string `json:"errmsg"`
  121. }
  122. func WxGetUserInfo(openId, accessToken string) (item *WxUserInfo, err error) {
  123. requestUrl := `https://api.weixin.qq.com/cgi-bin/user/info?access_token=%s&openid=%s`
  124. requestUrl = fmt.Sprintf(requestUrl, accessToken, openId)
  125. result, err := http.Get(requestUrl)
  126. if err != nil {
  127. return
  128. }
  129. fmt.Println("result:", string(result))
  130. utils.FileLog.Info("WxGetUserInfo:%s openId:%s,accessToken:%s ", string(result), openId, accessToken)
  131. utils.FileLog.Info("WxGetUserInfo Result:%s ", string(result))
  132. err = json.Unmarshal(result, &item)
  133. return
  134. }
  135. func GetWxTicket(accessToken string) (string, error) {
  136. Url := strings.Join([]string{"https://api.weixin.qq.com/cgi-bin/ticket/getticket",
  137. "?access_token=", accessToken,
  138. "&type=jsapi"}, "")
  139. infoBody, err := http.Get(Url)
  140. if err != nil {
  141. return "", err
  142. }
  143. atr := models.WxTicket{}
  144. err = json.Unmarshal(infoBody, &atr)
  145. fmt.Println("ticket result:", string(infoBody))
  146. if err != nil {
  147. return atr.Errmsg, err
  148. } else {
  149. return atr.Ticket, nil
  150. }
  151. }
  152. func GetWxSignature(ticket, url, noncestr string) (string, string, int64) {
  153. timestamp := time.Now().Unix()
  154. signStr := strings.Join([]string{"jsapi_ticket=", ticket,
  155. "&noncestr=", noncestr,
  156. "&timestamp=", strconv.FormatInt(timestamp, 10), "&url=", url}, "")
  157. signature := utils.Sha1(signStr)
  158. fmt.Println("signStr", signStr)
  159. return signature, noncestr, timestamp
  160. }
  161. type WxUserDetail struct {
  162. Unionid string
  163. Headimgurl string
  164. Nickname string
  165. }
  166. func FixUnionId() {
  167. accessToken := `41_fMVI8r0fzoDjGOsKjMBhVN9b0j34TWTFNHIX1cQ4IOJ8SnJ2Cjz6MhvHvtH9gU_ztKo-rRQopCgSjNSZB5z25KgB5o61822LDH1sR5ow_Rfoa_Ro6KwcdP2Gp_naNM0AwMGZaF5OMifMny-FFEXfAJAXWP`
  168. users, err := models.GetOpenIdAll()
  169. if err != nil {
  170. fmt.Println("GetOpenIdAll Err:" + err.Error())
  171. return
  172. }
  173. for _, v := range users {
  174. openId := v.OpenId
  175. fmt.Println("openId:", openId, v.UserId)
  176. userInfoUrl := `https://api.weixin.qq.com/cgi-bin/user/info?access_token=` + accessToken + `&openid=` + openId + `&lang=zh_CN`
  177. result, err := http.Get(userInfoUrl)
  178. fmt.Println("err:", err)
  179. utils.FileLog.Info("%s", string(result))
  180. item := new(WxUserDetail)
  181. err = json.Unmarshal(result, &item)
  182. if err != nil {
  183. fmt.Println("json.Unmarshal Err:" + err.Error())
  184. return
  185. }
  186. if item.Unionid != "" {
  187. err = models.ModifyWxUserUnionId(item.Unionid, v.UserId)
  188. if err != nil {
  189. fmt.Println("ModifyWxUserUnionId Err:" + err.Error())
  190. return
  191. }
  192. //return
  193. }
  194. }
  195. }
  196. // GetDefaultWxAccessToken 获取微信token
  197. func GetDefaultWxAccessToken() (accessToken string, err error, errMsg string) {
  198. // 微信配置信息
  199. conf, e := models.GetBusinessConf()
  200. if e != nil {
  201. errMsg = "获取配置信息失败, Err: " + e.Error()
  202. err = errors.New(errMsg)
  203. return
  204. }
  205. appId := ""
  206. appSecret := ""
  207. if v, ok := conf[models.BusinessConfWxAppId]; ok {
  208. appId = v
  209. }
  210. if v, ok := conf[models.BusinessConfWxAppSecret]; ok {
  211. appSecret = v
  212. }
  213. if appId == "" || appSecret == "" {
  214. errMsg = "获取微信公众号未配置"
  215. err = errors.New(errMsg)
  216. return
  217. }
  218. accessToken, err, errMsg = GetWxAccessToken(appId, appSecret)
  219. return
  220. }
  221. // GetWxAccessToken 获取微信token
  222. func GetWxAccessToken(wxAppId, wxAppSecret string) (accessToken string, err error, errMsg string) {
  223. redisKey := getRedisKeyByAppid(wxAppId)
  224. if redisKey == `` {
  225. errMsg = "未配置缓存key"
  226. err = errors.New(errMsg)
  227. return
  228. }
  229. accessToken, err = go_redis.RedisString(redisKey)
  230. //fmt.Println(err)
  231. //fmt.Println(accessToken)
  232. //if err != nil {
  233. // errMsg = "GetWxAccessToken Err:" + err.Error()
  234. // utils.FileLog.Info("获取Token失败,msg:" + errMsg)
  235. // return
  236. //}
  237. //取到数据后就直接返回了,没有后续了
  238. if accessToken != "" {
  239. return
  240. }
  241. //缓存中没有取到数据,那么需要去强制刷新新的accessToken
  242. return refreshWxAccessToken(wxAppId, wxAppSecret)
  243. }
  244. // refreshWxAccessToken 强制刷新微信token
  245. func refreshWxAccessToken(wxAppId, wxAppSecret string) (accessToken string, err error, errMsg string) {
  246. fmt.Println("强制刷新" + wxAppId + "微信token")
  247. defer func() {
  248. if errMsg != `` {
  249. utils.FileLog.Info(fmt.Sprintf("强制刷新%s微信token异常:%s", wxAppId, errMsg))
  250. }
  251. }()
  252. redisKey := getRedisKeyByAppid(wxAppId)
  253. if redisKey == `` {
  254. errMsg = "未配置缓存key"
  255. err = errors.New(errMsg)
  256. return
  257. }
  258. if wxAppSecret == "" {
  259. err = errors.New("缺少密钥信息")
  260. utils.FileLog.Info(fmt.Sprintf("获取Token失败, errMsg:%s", err.Error()))
  261. return
  262. }
  263. //调用微信官方接口获取新的accessToken
  264. wxAccessToken, tmpErr := WxGetToken(wxAppId, wxAppSecret)
  265. if tmpErr != nil {
  266. err = tmpErr
  267. errMsg = "通过微信接口获取accessToken失败 Err:" + err.Error()
  268. return
  269. }
  270. //如果没有token数据
  271. if wxAccessToken.AccessToken == "" {
  272. errMsg = "微信返回的accessToken异常: Err:" + wxAccessToken.Errmsg
  273. err = errors.New(errMsg)
  274. return
  275. }
  276. accessToken = wxAccessToken.AccessToken
  277. //如果是弘则研究的appid,那么需要更新mysql的accessToken
  278. if wxAppId == utils.WxAppId {
  279. expiresIn := time.Now().Add(time.Duration(wxAccessToken.ExpiresIn) * time.Second).Unix()
  280. err = models.ModifyAccessToken(wxAccessToken.AccessToken, expiresIn)
  281. if err != nil {
  282. errMsg = "更新mysql中的accessToken失败 Err:" + err.Error()
  283. return
  284. }
  285. }
  286. //更新redis的accessToken(过期时间提前十分钟)
  287. redisTimeExpire := time.Duration(wxAccessToken.ExpiresIn-600) * time.Second
  288. bo := go_redis.SetNX(redisKey, accessToken, redisTimeExpire)
  289. if !bo {
  290. errMsg = "更新redis中的accessToken失败"
  291. return
  292. }
  293. return
  294. }
  295. // 根据微信appid获取对应的缓存key
  296. func getRedisKeyByAppid(wxAppId string) (redisKey string) {
  297. switch wxAppId {
  298. case utils.WxAppId:
  299. redisKey = utils.CACHE_WX_ACCESS_TOKEN_HZ
  300. }
  301. return redisKey
  302. }