qrcode.go 3.1 KB

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