|
|
@@ -12,11 +12,15 @@ import (
|
|
|
|
|
|
const (
|
|
|
//AccessTokenURL 获取access_token的接口
|
|
|
- accessTokenURL = "https://api.weixin.qq.com/cgi-bin/token"
|
|
|
+ accessTokenURL = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s"
|
|
|
+ //AccessTokenURL 企业微信获取access_token的接口
|
|
|
+ workAccessTokenURL = "https://qyapi.weixin.qq.com/cgi-bin/gettoken?corpid=%s&corpsecret=%s"
|
|
|
//CacheKeyOfficialAccountPrefix 微信公众号cache key前缀
|
|
|
CacheKeyOfficialAccountPrefix = "gowechat_officialaccount_"
|
|
|
//CacheKeyMiniProgramPrefix 小程序cache key前缀
|
|
|
CacheKeyMiniProgramPrefix = "gowechat_miniprogram_"
|
|
|
+ //CacheKeyWorkPrefix 企业微信cache key前缀
|
|
|
+ CacheKeyWorkPrefix = "gowechat_work_"
|
|
|
)
|
|
|
|
|
|
//DefaultAccessToken 默认AccessToken 获取
|
|
|
@@ -65,7 +69,58 @@ func (ak *DefaultAccessToken) GetAccessToken() (accessToken string, err error) {
|
|
|
|
|
|
//cache失效,从微信服务器获取
|
|
|
var resAccessToken ResAccessToken
|
|
|
- resAccessToken, err = GetTokenFromServer(ak.appID, ak.appSecret)
|
|
|
+ resAccessToken, err = GetTokenFromServer(fmt.Sprintf(accessTokenURL, ak.appID, ak.appSecret))
|
|
|
+ if err != nil {
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ expires := resAccessToken.ExpiresIn - 1500
|
|
|
+ err = ak.cache.Set(accessTokenCacheKey, resAccessToken.AccessToken, time.Duration(expires)*time.Second)
|
|
|
+ if err != nil {
|
|
|
+ return
|
|
|
+ }
|
|
|
+ accessToken = resAccessToken.AccessToken
|
|
|
+ return
|
|
|
+}
|
|
|
+
|
|
|
+//WorkAccessToken 企业微信AccessToken 获取
|
|
|
+type WorkAccessToken struct {
|
|
|
+ CorpID string
|
|
|
+ CorpSecret string
|
|
|
+ cacheKeyPrefix string
|
|
|
+ cache cache.Cache
|
|
|
+ accessTokenLock *sync.Mutex
|
|
|
+}
|
|
|
+
|
|
|
+//NewWorkAccessToken new WorkAccessToken
|
|
|
+func NewWorkAccessToken(corpID, corpSecret, cacheKeyPrefix string, cache cache.Cache) AccessTokenHandle {
|
|
|
+ if cache == nil {
|
|
|
+ panic("cache the not exist")
|
|
|
+ }
|
|
|
+ return &WorkAccessToken{
|
|
|
+ CorpID: corpID,
|
|
|
+ CorpSecret: corpSecret,
|
|
|
+ cache: cache,
|
|
|
+ cacheKeyPrefix: cacheKeyPrefix,
|
|
|
+ accessTokenLock: new(sync.Mutex),
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+//GetAccessToken 企业微信获取access_token,先从cache中获取,没有则从服务端获取
|
|
|
+func (ak *WorkAccessToken) GetAccessToken() (accessToken string, err error) {
|
|
|
+ //加上lock,是为了防止在并发获取token时,cache刚好失效,导致从微信服务器上获取到不同token
|
|
|
+ ak.accessTokenLock.Lock()
|
|
|
+ defer ak.accessTokenLock.Unlock()
|
|
|
+ accessTokenCacheKey := fmt.Sprintf("%s_access_token_%s", ak.cacheKeyPrefix, ak.CorpID)
|
|
|
+ val := ak.cache.Get(accessTokenCacheKey)
|
|
|
+ if val != nil {
|
|
|
+ accessToken = val.(string)
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ //cache失效,从微信服务器获取
|
|
|
+ var resAccessToken ResAccessToken
|
|
|
+ resAccessToken, err = GetTokenFromServer(fmt.Sprintf(workAccessTokenURL, ak.CorpID, ak.CorpSecret))
|
|
|
if err != nil {
|
|
|
return
|
|
|
}
|
|
|
@@ -80,8 +135,7 @@ func (ak *DefaultAccessToken) GetAccessToken() (accessToken string, err error) {
|
|
|
}
|
|
|
|
|
|
//GetTokenFromServer 强制从微信服务器获取token
|
|
|
-func GetTokenFromServer(appID, appSecret string) (resAccessToken ResAccessToken, err error) {
|
|
|
- url := fmt.Sprintf("%s?grant_type=client_credential&appid=%s&secret=%s", accessTokenURL, appID, appSecret)
|
|
|
+func GetTokenFromServer(url string) (resAccessToken ResAccessToken, err error) {
|
|
|
var body []byte
|
|
|
body, err = util.HTTPGet(url)
|
|
|
if err != nil {
|
|
|
@@ -91,7 +145,7 @@ func GetTokenFromServer(appID, appSecret string) (resAccessToken ResAccessToken,
|
|
|
if err != nil {
|
|
|
return
|
|
|
}
|
|
|
- if resAccessToken.ErrMsg != "" {
|
|
|
+ if resAccessToken.ErrCode != 0 {
|
|
|
err = fmt.Errorf("get access_token error : errcode=%v , errormsg=%v", resAccessToken.ErrCode, resAccessToken.ErrMsg)
|
|
|
return
|
|
|
}
|