default_access_token.go 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284
  1. package credential
  2. import (
  3. "context"
  4. "encoding/json"
  5. "fmt"
  6. "sync"
  7. "time"
  8. "github.com/silenceper/wechat/v2/cache"
  9. "github.com/silenceper/wechat/v2/util"
  10. )
  11. const (
  12. // accessTokenURL 获取access_token的接口
  13. accessTokenURL = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s"
  14. // stableAccessTokenURL 获取稳定版access_token的接口
  15. stableAccessTokenURL = "https://api.weixin.qq.com/cgi-bin/stable_token"
  16. // workAccessTokenURL 企业微信获取access_token的接口
  17. workAccessTokenURL = "https://qyapi.weixin.qq.com/cgi-bin/gettoken?corpid=%s&corpsecret=%s"
  18. // CacheKeyOfficialAccountPrefix 微信公众号cache key前缀
  19. CacheKeyOfficialAccountPrefix = "gowechat_officialaccount_"
  20. // CacheKeyMiniProgramPrefix 小程序cache key前缀
  21. CacheKeyMiniProgramPrefix = "gowechat_miniprogram_"
  22. // CacheKeyWorkPrefix 企业微信cache key前缀
  23. CacheKeyWorkPrefix = "gowechat_work_"
  24. )
  25. // DefaultAccessToken 默认AccessToken 获取
  26. type DefaultAccessToken struct {
  27. appID string
  28. appSecret string
  29. cacheKeyPrefix string
  30. cache cache.Cache
  31. accessTokenLock *sync.Mutex
  32. }
  33. // NewDefaultAccessToken new DefaultAccessToken
  34. func NewDefaultAccessToken(appID, appSecret, cacheKeyPrefix string, cache cache.Cache) AccessTokenContextHandle {
  35. if cache == nil {
  36. panic("cache is ineed")
  37. }
  38. return &DefaultAccessToken{
  39. appID: appID,
  40. appSecret: appSecret,
  41. cache: cache,
  42. cacheKeyPrefix: cacheKeyPrefix,
  43. accessTokenLock: new(sync.Mutex),
  44. }
  45. }
  46. // ResAccessToken struct
  47. type ResAccessToken struct {
  48. util.CommonError
  49. AccessToken string `json:"access_token"`
  50. ExpiresIn int64 `json:"expires_in"`
  51. }
  52. // GetAccessToken 获取access_token,先从cache中获取,没有则从服务端获取
  53. func (ak *DefaultAccessToken) GetAccessToken() (accessToken string, err error) {
  54. return ak.GetAccessTokenContext(context.Background())
  55. }
  56. // GetAccessTokenContext 获取access_token,先从cache中获取,没有则从服务端获取
  57. func (ak *DefaultAccessToken) GetAccessTokenContext(ctx context.Context) (accessToken string, err error) {
  58. // 先从cache中取
  59. accessTokenCacheKey := fmt.Sprintf("%s_access_token_%s", ak.cacheKeyPrefix, ak.appID)
  60. if val := ak.cache.Get(accessTokenCacheKey); val != nil {
  61. if accessToken = val.(string); accessToken != "" {
  62. return
  63. }
  64. }
  65. // 加上lock,是为了防止在并发获取token时,cache刚好失效,导致从微信服务器上获取到不同token
  66. ak.accessTokenLock.Lock()
  67. defer ak.accessTokenLock.Unlock()
  68. // 双检,防止重复从微信服务器获取
  69. if val := ak.cache.Get(accessTokenCacheKey); val != nil {
  70. if accessToken = val.(string); accessToken != "" {
  71. return
  72. }
  73. }
  74. // cache失效,从微信服务器获取
  75. var resAccessToken ResAccessToken
  76. if resAccessToken, err = GetTokenFromServerContext(ctx, fmt.Sprintf(accessTokenURL, ak.appID, ak.appSecret)); err != nil {
  77. return
  78. }
  79. expires := resAccessToken.ExpiresIn - 1500
  80. err = ak.cache.Set(accessTokenCacheKey, resAccessToken.AccessToken, time.Duration(expires)*time.Second)
  81. accessToken = resAccessToken.AccessToken
  82. return
  83. }
  84. // StableAccessToken 获取稳定版接口调用凭据(与getAccessToken获取的调用凭证完全隔离,互不影响)
  85. // 不强制更新access_token,可用于不同环境不同服务而不需要分布式锁以及公用缓存,避免access_token争抢
  86. // https://developers.weixin.qq.com/miniprogram/dev/OpenApiDoc/mp-access-token/getStableAccessToken.html
  87. type StableAccessToken struct {
  88. appID string
  89. appSecret string
  90. cacheKeyPrefix string
  91. cache cache.Cache
  92. accessTokenLock *sync.Mutex
  93. }
  94. // NewStableAccessToken new StableAccessToken
  95. func NewStableAccessToken(appID, appSecret, cacheKeyPrefix string, cache cache.Cache) AccessTokenContextHandle {
  96. if cache == nil {
  97. panic("cache is need")
  98. }
  99. return &StableAccessToken{
  100. appID: appID,
  101. appSecret: appSecret,
  102. cache: cache,
  103. cacheKeyPrefix: cacheKeyPrefix,
  104. accessTokenLock: new(sync.Mutex),
  105. }
  106. }
  107. // GetAccessToken 获取access_token,先从cache中获取,没有则从服务端获取
  108. func (ak *StableAccessToken) GetAccessToken() (accessToken string, err error) {
  109. return ak.GetAccessTokenContext(context.Background())
  110. }
  111. // GetAccessTokenContext 获取access_token,先从cache中获取,没有则从服务端获取
  112. func (ak *StableAccessToken) GetAccessTokenContext(ctx context.Context) (accessToken string, err error) {
  113. // 先从cache中取
  114. accessTokenCacheKey := fmt.Sprintf("%s_stable_access_token_%s", ak.cacheKeyPrefix, ak.appID)
  115. if val := ak.cache.Get(accessTokenCacheKey); val != nil {
  116. if accessToken = val.(string); accessToken != "" {
  117. return
  118. }
  119. }
  120. // 加上lock,是为了防止在并发获取token时,cache刚好失效,导致从微信服务器上获取到不同token
  121. ak.accessTokenLock.Lock()
  122. defer ak.accessTokenLock.Unlock()
  123. // 双检,防止重复从微信服务器获取
  124. if val := ak.cache.Get(accessTokenCacheKey); val != nil {
  125. if accessToken = val.(string); accessToken != "" {
  126. return
  127. }
  128. }
  129. // cache失效,从微信服务器获取
  130. var resAccessToken ResAccessToken
  131. resAccessToken, err = ak.GetAccessTokenDirectly(ctx, false)
  132. if err != nil {
  133. return
  134. }
  135. expires := resAccessToken.ExpiresIn - 300
  136. err = ak.cache.Set(accessTokenCacheKey, resAccessToken.AccessToken, time.Duration(expires)*time.Second)
  137. accessToken = resAccessToken.AccessToken
  138. return
  139. }
  140. // GetAccessTokenDirectly 从微信获取access_token
  141. func (ak *StableAccessToken) GetAccessTokenDirectly(ctx context.Context, forceRefresh bool) (resAccessToken ResAccessToken, err error) {
  142. b, err := util.PostJSONContext(ctx, stableAccessTokenURL, map[string]interface{}{
  143. "grant_type": "client_credential",
  144. "appid": ak.appID,
  145. "secret": ak.appSecret,
  146. "force_refresh": forceRefresh,
  147. })
  148. if err != nil {
  149. return
  150. }
  151. if err = json.Unmarshal(b, &resAccessToken); err != nil {
  152. return
  153. }
  154. if resAccessToken.ErrCode != 0 {
  155. err = fmt.Errorf("get stable access_token error : errcode=%v , errormsg=%v", resAccessToken.ErrCode, resAccessToken.ErrMsg)
  156. return
  157. }
  158. return
  159. }
  160. // WorkAccessToken 企业微信AccessToken 获取
  161. type WorkAccessToken struct {
  162. CorpID string
  163. CorpSecret string
  164. AgentID string // 可选,用于区分不同应用
  165. cacheKeyPrefix string
  166. cache cache.Cache
  167. accessTokenLock *sync.Mutex
  168. }
  169. // NewWorkAccessToken new WorkAccessToken (保持向后兼容)
  170. func NewWorkAccessToken(corpID, corpSecret, agentID, cacheKeyPrefix string, cache cache.Cache) AccessTokenContextHandle {
  171. // 调用新方法,保持兼容性
  172. return NewWorkAccessTokenWithAgentID(corpID, corpSecret, agentID, cacheKeyPrefix, cache)
  173. }
  174. // NewWorkAccessTokenWithAgentID new WorkAccessToken with agentID
  175. func NewWorkAccessTokenWithAgentID(corpID, corpSecret, agentID, cacheKeyPrefix string, cache cache.Cache) AccessTokenContextHandle {
  176. if cache == nil {
  177. panic("cache is needed")
  178. }
  179. return &WorkAccessToken{
  180. CorpID: corpID,
  181. CorpSecret: corpSecret,
  182. AgentID: agentID,
  183. cache: cache,
  184. cacheKeyPrefix: cacheKeyPrefix,
  185. accessTokenLock: new(sync.Mutex),
  186. }
  187. }
  188. // GetAccessToken 企业微信获取access_token,先从cache中获取,没有则从服务端获取
  189. func (ak *WorkAccessToken) GetAccessToken() (accessToken string, err error) {
  190. return ak.GetAccessTokenContext(context.Background())
  191. }
  192. // GetAccessTokenContext 企业微信获取access_token,先从cache中获取,没有则从服务端获取
  193. func (ak *WorkAccessToken) GetAccessTokenContext(ctx context.Context) (accessToken string, err error) {
  194. // 加上lock,是为了防止在并发获取token时,cache刚好失效,导致从微信服务器上获取到不同token
  195. ak.accessTokenLock.Lock()
  196. defer ak.accessTokenLock.Unlock()
  197. // 构建缓存key
  198. var accessTokenCacheKey string
  199. if ak.AgentID != "" {
  200. // 如果设置了AgentID,使用新的key格式
  201. accessTokenCacheKey = fmt.Sprintf("%s_access_token_%s_%s", ak.cacheKeyPrefix, ak.CorpID, ak.AgentID)
  202. } else {
  203. // 兼容历史版本的key格式
  204. accessTokenCacheKey = fmt.Sprintf("%s_access_token_%s", ak.cacheKeyPrefix, ak.CorpID)
  205. }
  206. val := ak.cache.Get(accessTokenCacheKey)
  207. if val != nil {
  208. accessToken = val.(string)
  209. return
  210. }
  211. // cache失效,从微信服务器获取
  212. var resAccessToken ResAccessToken
  213. resAccessToken, err = GetTokenFromServerContext(ctx, fmt.Sprintf(workAccessTokenURL, ak.CorpID, ak.CorpSecret))
  214. if err != nil {
  215. return
  216. }
  217. expires := resAccessToken.ExpiresIn - 1500
  218. err = ak.cache.Set(accessTokenCacheKey, resAccessToken.AccessToken, time.Duration(expires)*time.Second)
  219. if err != nil {
  220. return
  221. }
  222. accessToken = resAccessToken.AccessToken
  223. return
  224. }
  225. // GetTokenFromServer 强制从微信服务器获取token
  226. func GetTokenFromServer(url string) (resAccessToken ResAccessToken, err error) {
  227. return GetTokenFromServerContext(context.Background(), url)
  228. }
  229. // GetTokenFromServerContext 强制从微信服务器获取token
  230. func GetTokenFromServerContext(ctx context.Context, url string) (resAccessToken ResAccessToken, err error) {
  231. var body []byte
  232. body, err = util.HTTPGetContext(ctx, url)
  233. if err != nil {
  234. return
  235. }
  236. err = json.Unmarshal(body, &resAccessToken)
  237. if err != nil {
  238. return
  239. }
  240. if resAccessToken.ErrCode != 0 {
  241. err = fmt.Errorf("get access_token error : errcode=%v , errormsg=%v", resAccessToken.ErrCode, resAccessToken.ErrMsg)
  242. return
  243. }
  244. return
  245. }