Skip to content

Commit 9c87d1c

Browse files
authored
feat:企业微信客户端API JS-SDK wx.config 和 wx.agentConfig 方法权限签名 (#817)
* feat: enhance WorkAccessToken to include AgentID for improved token management - Added AgentID field to WorkAccessToken struct. - Updated NewWorkAccessToken function to accept AgentID as a parameter. - Modified access token cache key to incorporate AgentID, ensuring unique cache entries per agent. This change improves the handling of access tokens in a multi-agent environment. * refactor: enhance WorkAccessToken to improve cache key handling - Updated the AgentID field in WorkAccessToken struct to clarify its optional nature for distinguishing applications. - Modified the access token cache key construction to support both new and legacy formats based on the presence of AgentID. - Added comments for better understanding of the cache key logic and its compatibility with historical versions. This change improves the flexibility and clarity of access token management in multi-agent scenarios. * feat(work): add JsSdk method for JavaScript SDK integration - Introduced a new JsSdk method in the Work struct to facilitate the creation of a Js instance. - This addition enhances the functionality of the Work module by enabling JavaScript SDK support. This change improves the integration capabilities for developers working with the WeChat Work API. * fix gofmt
1 parent 71c8ab5 commit 9c87d1c

File tree

4 files changed

+198
-0
lines changed

4 files changed

+198
-0
lines changed

credential/default_access_token.go

+1
Original file line numberDiff line numberDiff line change
@@ -229,6 +229,7 @@ func (ak *WorkAccessToken) GetAccessTokenContext(ctx context.Context) (accessTok
229229

230230
// 构建缓存key
231231
var accessTokenCacheKey string
232+
232233
if ak.AgentID != "" {
233234
// 如果设置了AgentID,使用新的key格式
234235
accessTokenCacheKey = fmt.Sprintf("%s_access_token_%s_%s", ak.cacheKeyPrefix, ak.CorpID, ak.AgentID)

credential/work_js_ticket.go

+118
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,118 @@
1+
package credential
2+
3+
import (
4+
"encoding/json"
5+
"fmt"
6+
"sync"
7+
"time"
8+
9+
"github.com/silenceper/wechat/v2/cache"
10+
"github.com/silenceper/wechat/v2/util"
11+
)
12+
13+
// TicketType ticket类型
14+
type TicketType int
15+
16+
const (
17+
// TicketTypeCorpJs 企业jsapi ticket
18+
TicketTypeCorpJs TicketType = iota
19+
// TicketTypeAgentJs 应用jsapi ticket
20+
TicketTypeAgentJs
21+
)
22+
23+
// 企业微信相关的 ticket URL
24+
const (
25+
// 企业微信 jsapi ticket
26+
getWorkJsTicketURL = "https://qyapi.weixin.qq.com/cgi-bin/get_jsapi_ticket?access_token=%s"
27+
// 企业微信应用 jsapi ticket
28+
getWorkAgentJsTicketURL = "https://qyapi.weixin.qq.com/cgi-bin/ticket/get?access_token=%s&type=agent_config"
29+
)
30+
31+
// WorkJsTicket 企业微信js ticket获取
32+
type WorkJsTicket struct {
33+
corpID string
34+
agentID string
35+
cacheKeyPrefix string
36+
cache cache.Cache
37+
jsAPITicketLock *sync.Mutex
38+
}
39+
40+
// NewWorkJsTicket new WorkJsTicket
41+
func NewWorkJsTicket(corpID, agentID, cacheKeyPrefix string, cache cache.Cache) *WorkJsTicket {
42+
return &WorkJsTicket{
43+
corpID: corpID,
44+
agentID: agentID,
45+
cache: cache,
46+
cacheKeyPrefix: cacheKeyPrefix,
47+
jsAPITicketLock: new(sync.Mutex),
48+
}
49+
}
50+
51+
// GetTicket 根据类型获取相应的jsapi_ticket
52+
func (js *WorkJsTicket) GetTicket(accessToken string, ticketType TicketType) (ticketStr string, err error) {
53+
var cacheKey string
54+
switch ticketType {
55+
case TicketTypeCorpJs:
56+
cacheKey = fmt.Sprintf("%s_corp_jsapi_ticket_%s", js.cacheKeyPrefix, js.corpID)
57+
case TicketTypeAgentJs:
58+
if js.agentID == "" {
59+
err = fmt.Errorf("agentID is empty")
60+
return
61+
}
62+
cacheKey = fmt.Sprintf("%s_agent_jsapi_ticket_%s_%s", js.cacheKeyPrefix, js.corpID, js.agentID)
63+
default:
64+
err = fmt.Errorf("unsupported ticket type: %v", ticketType)
65+
return
66+
}
67+
68+
if val := js.cache.Get(cacheKey); val != nil {
69+
return val.(string), nil
70+
}
71+
72+
js.jsAPITicketLock.Lock()
73+
defer js.jsAPITicketLock.Unlock()
74+
75+
// 双检,防止重复从微信服务器获取
76+
if val := js.cache.Get(cacheKey); val != nil {
77+
return val.(string), nil
78+
}
79+
80+
var ticket ResTicket
81+
ticket, err = js.getTicketFromServer(accessToken, ticketType)
82+
if err != nil {
83+
return
84+
}
85+
expires := ticket.ExpiresIn - 1500
86+
err = js.cache.Set(cacheKey, ticket.Ticket, time.Duration(expires)*time.Second)
87+
ticketStr = ticket.Ticket
88+
return
89+
}
90+
91+
// getTicketFromServer 从服务器中获取ticket
92+
func (js *WorkJsTicket) getTicketFromServer(accessToken string, ticketType TicketType) (ticket ResTicket, err error) {
93+
var url string
94+
switch ticketType {
95+
case TicketTypeCorpJs:
96+
url = fmt.Sprintf(getWorkJsTicketURL, accessToken)
97+
case TicketTypeAgentJs:
98+
url = fmt.Sprintf(getWorkAgentJsTicketURL, accessToken)
99+
default:
100+
err = fmt.Errorf("unsupported ticket type: %v", ticketType)
101+
return
102+
}
103+
104+
var response []byte
105+
response, err = util.HTTPGet(url)
106+
if err != nil {
107+
return
108+
}
109+
err = json.Unmarshal(response, &ticket)
110+
if err != nil {
111+
return
112+
}
113+
if ticket.ErrCode != 0 {
114+
err = fmt.Errorf("getTicket Error : errcode=%d , errmsg=%s", ticket.ErrCode, ticket.ErrMsg)
115+
return
116+
}
117+
return
118+
}

work/jsapi/jsapi.go

+73
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
package jsapi
2+
3+
import (
4+
"strconv"
5+
6+
"github.com/silenceper/wechat/v2/credential"
7+
"github.com/silenceper/wechat/v2/util"
8+
"github.com/silenceper/wechat/v2/work/context"
9+
)
10+
11+
// Js struct
12+
type Js struct {
13+
*context.Context
14+
jsTicket *credential.WorkJsTicket
15+
}
16+
17+
// NewJs init
18+
func NewJs(context *context.Context) *Js {
19+
js := new(Js)
20+
js.Context = context
21+
js.jsTicket = credential.NewWorkJsTicket(
22+
context.Config.CorpID,
23+
context.Config.AgentID,
24+
credential.CacheKeyWorkPrefix,
25+
context.Cache,
26+
)
27+
return js
28+
}
29+
30+
// Config 返回给用户使用的配置
31+
type Config struct {
32+
Timestamp int64 `json:"timestamp"`
33+
NonceStr string `json:"nonce_str"`
34+
Signature string `json:"signature"`
35+
}
36+
37+
// GetConfig 获取企业微信JS配置 https://developer.work.weixin.qq.com/document/path/90514
38+
func (js *Js) GetConfig(uri string) (config *Config, err error) {
39+
config = new(Config)
40+
var accessToken string
41+
accessToken, err = js.GetAccessToken()
42+
if err != nil {
43+
return
44+
}
45+
var ticketStr string
46+
ticketStr, err = js.jsTicket.GetTicket(accessToken, credential.TicketTypeCorpJs)
47+
if err != nil {
48+
return
49+
}
50+
config.NonceStr = util.RandomStr(16)
51+
config.Timestamp = util.GetCurrTS()
52+
config.Signature = util.Signature(ticketStr, config.NonceStr, strconv.FormatInt(config.Timestamp, 10), uri)
53+
return
54+
}
55+
56+
// GetAgentConfig 获取企业微信应用JS配置 https://developer.work.weixin.qq.com/document/path/94313
57+
func (js *Js) GetAgentConfig(uri string) (config *Config, err error) {
58+
config = new(Config)
59+
var accessToken string
60+
accessToken, err = js.GetAccessToken()
61+
if err != nil {
62+
return
63+
}
64+
var ticketStr string
65+
ticketStr, err = js.jsTicket.GetTicket(accessToken, credential.TicketTypeAgentJs)
66+
if err != nil {
67+
return
68+
}
69+
config.NonceStr = util.RandomStr(16)
70+
config.Timestamp = util.GetCurrTS()
71+
config.Signature = util.Signature(ticketStr, config.NonceStr, strconv.FormatInt(config.Timestamp, 10), uri)
72+
return
73+
}

work/work.go

+6
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ import (
99
"github.com/silenceper/wechat/v2/work/context"
1010
"github.com/silenceper/wechat/v2/work/externalcontact"
1111
"github.com/silenceper/wechat/v2/work/invoice"
12+
"github.com/silenceper/wechat/v2/work/jsapi"
1213
"github.com/silenceper/wechat/v2/work/kf"
1314
"github.com/silenceper/wechat/v2/work/material"
1415
"github.com/silenceper/wechat/v2/work/message"
@@ -52,6 +53,11 @@ func (wk *Work) GetKF() (*kf.Client, error) {
5253
return kf.NewClient(wk.ctx.Config)
5354
}
5455

56+
// JsSdk get JsSdk
57+
func (wk *Work) JsSdk() *jsapi.Js {
58+
return jsapi.NewJs(wk.ctx)
59+
}
60+
5561
// GetExternalContact get external_contact
5662
func (wk *Work) GetExternalContact() *externalcontact.Client {
5763
return externalcontact.NewClient(wk.ctx)

0 commit comments

Comments
 (0)