qrcode.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. package service
  2. import (
  3. "bytes"
  4. "context"
  5. "encoding/json"
  6. "errors"
  7. "fmt"
  8. "io/ioutil"
  9. "net/http"
  10. "strconv"
  11. "strings"
  12. "time"
  13. "youngee_b_api/consts"
  14. "youngee_b_api/model/http_model"
  15. "youngee_b_api/model/system_model"
  16. "youngee_b_api/redis"
  17. )
  18. const (
  19. accessTokenUrlFormat = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s"
  20. qrCodeUrlFormat = "https://api.weixin.qq.com/wxa/getwxacodeunlimit?access_token=%s"
  21. wxAccesssTokenKey = "wx_access_token"
  22. )
  23. var QrCode *qrcode
  24. type qrcode struct {
  25. sessionTTL time.Duration
  26. }
  27. func QrCodeInit(config *system_model.Session) {
  28. qrCode := new(qrcode)
  29. qrCode.sessionTTL = 5 * time.Minute
  30. QrCode = qrCode
  31. }
  32. // getAndCacheWxAccessToken 获取并缓存微信的access token
  33. func getAndCacheWxAccessToken(ctx context.Context) (string, error) {
  34. appId := "wxac396a3be7a16844"
  35. // secret := "c82ae9e75b4ed7d8022db5bda5371892"
  36. secret := "abbf27d46c46212c86e60f2ed3c534ee"
  37. url := fmt.Sprintf(accessTokenUrlFormat, appId, secret)
  38. resp, err := http.Get(url)
  39. if err != nil {
  40. return "", errors.New("request access token failed")
  41. }
  42. defer resp.Body.Close()
  43. // 解析微信服务端返回的信息
  44. var accessTokenRes http_model.WxAccessTokenResponse
  45. decoder := json.NewDecoder(resp.Body)
  46. if err = decoder.Decode(&accessTokenRes); err != nil {
  47. return "", errors.New("decode wx response failed")
  48. }
  49. if accessTokenRes.Errcode != 0 {
  50. return "", errors.New("request access token failed")
  51. }
  52. // 缓存获取的access token,比微信返回的有效时间短5分钟失效
  53. err = redis.Set(ctx, wxAccesssTokenKey, accessTokenRes.AccessToken, QrCode.sessionTTL)
  54. if err != nil {
  55. return "", err
  56. }
  57. return accessTokenRes.AccessToken, nil
  58. }
  59. func (q *qrcode) GetWxQrCode(ctx context.Context, req *http_model.GetWxQRCodeRequest) (*http_model.GetWxQRCodeData, error) {
  60. // 获取access_token
  61. accessToken, err := redis.Get(ctx, wxAccesssTokenKey)
  62. if err != nil && err != consts.RedisNil {
  63. fmt.Printf("err: %+v\n", err)
  64. return nil, err
  65. }
  66. if accessToken == "" {
  67. // 如果没有缓存的access token 则获取并缓存
  68. accessToken, err = getAndCacheWxAccessToken(ctx)
  69. if err != nil {
  70. fmt.Printf("err: %+v\n", err)
  71. return nil, err
  72. }
  73. }
  74. var reqPath string
  75. var reqData string
  76. if req.SelectionId != "" {
  77. reqPath = "pageCommerce/taskDetail"
  78. reqData = req.SelectionId
  79. }
  80. if req.SProjectId != 0 || req.ProjectId != "" {
  81. reqPath = "pageRecommend/recommendDetail"
  82. if req.SProjectId != 0 {
  83. reqData = "0" + "-" + strconv.Itoa(req.SProjectId)
  84. } else {
  85. reqData = req.ProjectId + "-" + "0"
  86. }
  87. }
  88. if req.SLocalId != 0 || req.LocalId != "" {
  89. reqPath = "pageLife/lifeDetail"
  90. if req.SLocalId != 0 {
  91. reqData = "0" + "-" + strconv.Itoa(req.SLocalId)
  92. } else {
  93. reqData = req.LocalId + "-" + "0"
  94. }
  95. }
  96. fmt.Printf("reqData: %+v\n", reqData)
  97. qrRequest := http_model.WxQrCodeRequest{
  98. Scene: reqData,
  99. Page: reqPath,
  100. Width: 430,
  101. CheckPath: false,
  102. EnvVersion: "release",
  103. }
  104. jsonBody, err := json.Marshal(qrRequest)
  105. if err != nil {
  106. fmt.Printf("err: %+v\n", err)
  107. return nil, err
  108. }
  109. qrCodeUrl := fmt.Sprintf(qrCodeUrlFormat, accessToken)
  110. resp, err := http.Post(qrCodeUrl, "application/json; charset=utf8", bytes.NewReader(jsonBody))
  111. if err != nil {
  112. fmt.Printf("err: %+v\n", err)
  113. return nil, err
  114. }
  115. defer resp.Body.Close()
  116. var qrcodeBytes []byte
  117. switch header := resp.Header.Get("content-Type"); {
  118. case strings.HasPrefix(header, "application/json"):
  119. // 如果返回的是json结构,说明发生错误
  120. var qrResponse *http_model.WxQrCodeResponse
  121. decoder := json.NewDecoder(resp.Body)
  122. if err = decoder.Decode(&qrResponse); err != nil {
  123. fmt.Printf("decoder: %+v\nerr: %+v", decoder, err)
  124. return nil, err
  125. }
  126. fmt.Printf("qrResponse: %+v\ndecoder: %+v\n", qrResponse, decoder)
  127. return nil, nil
  128. case strings.HasPrefix(header, "image"):
  129. qrcodeBytes, err = ioutil.ReadAll(resp.Body)
  130. if err != nil {
  131. fmt.Printf("qrcodeBytes: %+v\nerr: %+v", qrcodeBytes, err)
  132. return nil, err
  133. }
  134. }
  135. data := http_model.GetWxQRCodeData{
  136. QrCodeBytes: qrcodeBytes,
  137. }
  138. //fmt.Printf("data: %+v\n", data)
  139. return &data, nil
  140. }