access_token.go 2.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. package context
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "sync"
  6. "time"
  7. "github.com/silenceper/wechat/util"
  8. )
  9. const (
  10. //AccessTokenURL 获取access_token的接口
  11. AccessTokenURL = "https://api.weixin.qq.com/cgi-bin/token"
  12. )
  13. //ResAccessToken struct
  14. type ResAccessToken struct {
  15. util.CommonError
  16. AccessToken string `json:"access_token"`
  17. ExpiresIn int64 `json:"expires_in"`
  18. }
  19. type CustomGetAccessToken func(ctx *Context) (accessToken string, err error)
  20. //SetAccessTokenLock 设置读写锁(一个appID一个读写锁)
  21. func (ctx *Context) SetAccessTokenLock(l *sync.RWMutex) {
  22. ctx.accessTokenLock = l
  23. }
  24. //SetGetAccessTokenFunc 设置自定义获取accessToken的方式, 需要自己实现缓存
  25. func (ctx *Context) SetGetAccessTokenFunc(f CustomGetAccessToken) {
  26. ctx.accessTokenFunc = f
  27. }
  28. //GetAccessToken 获取access_token
  29. func (ctx *Context) GetAccessToken() (accessToken string, err error) {
  30. ctx.accessTokenLock.Lock()
  31. defer ctx.accessTokenLock.Unlock()
  32. if ctx.accessTokenFunc != nil {
  33. return ctx.accessTokenFunc(ctx)
  34. }
  35. accessTokenCacheKey := fmt.Sprintf("access_token_%s", ctx.AppID)
  36. val := ctx.Cache.Get(accessTokenCacheKey)
  37. if val != nil {
  38. accessToken = val.(string)
  39. return
  40. }
  41. //从微信服务器获取
  42. var resAccessToken ResAccessToken
  43. resAccessToken, err = ctx.GetAccessTokenFromServer()
  44. if err != nil {
  45. return
  46. }
  47. accessToken = resAccessToken.AccessToken
  48. return
  49. }
  50. //GetAccessTokenFromServer 强制从微信服务器获取token
  51. func (ctx *Context) GetAccessTokenFromServer() (resAccessToken ResAccessToken, err error) {
  52. url := fmt.Sprintf("%s?grant_type=client_credential&appid=%s&secret=%s", AccessTokenURL, ctx.AppID, ctx.AppSecret)
  53. var body []byte
  54. body, err = util.HTTPGet(url)
  55. if err != nil {
  56. return
  57. }
  58. err = json.Unmarshal(body, &resAccessToken)
  59. if err != nil {
  60. return
  61. }
  62. if resAccessToken.ErrMsg != "" {
  63. err = fmt.Errorf("get access_token error : errcode=%v , errormsg=%v", resAccessToken.ErrCode, resAccessToken.ErrMsg)
  64. return
  65. }
  66. accessTokenCacheKey := fmt.Sprintf("access_token_%s", ctx.AppID)
  67. expires := resAccessToken.ExpiresIn - 1500
  68. err = ctx.Cache.Set(accessTokenCacheKey, resAccessToken.AccessToken, time.Duration(expires)*time.Second)
  69. return
  70. }