wxapi.go 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. package wxapi
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "io/ioutil"
  8. "net/http"
  9. "strings"
  10. "sync"
  11. "time"
  12. )
  13. const (
  14. GetTokenUrl = "https://qyapi.weixin.qq.com/cgi-bin/gettoken?corpid=ID&corpsecret=SECRET"
  15. GetServiceUrl = "https://qyapi.weixin.qq.com/cgi-bin/kf/account/list?access_token=ACCESS_TOKEN"
  16. GetServicerUrl = "https://qyapi.weixin.qq.com/cgi-bin/kf/servicer/list?access_token=ACCESS_TOKEN&open_kfid=OPEN_KFID"
  17. GetStaffUrl = "https://qyapi.weixin.qq.com/cgi-bin/user/get?access_token=ACCESS_TOKEN&userid=USERID"
  18. GetCustomerUrl = "https://qyapi.weixin.qq.com/cgi-bin/kf/customer/batchget?access_token=ACCESS_TOKEN"
  19. GetMsgUrl = "https://qyapi.weixin.qq.com/cgi-bin/kf/sync_msg?access_token=ACCESS_TOKEN"
  20. GetSessionStateUrl = "https://qyapi.weixin.qq.com/cgi-bin/kf/service_state/get?access_token=ACCESS_TOKEN"
  21. UpdateSessionStateUrl = "https://qyapi.weixin.qq.com/cgi-bin/kf/service_state/trans?access_token=ACCESS_TOKEN"
  22. )
  23. // 自定义封装企业微信的API接口
  24. type WxApi struct {
  25. lock sync.Mutex
  26. corpid string
  27. corpsecret string
  28. accessToken string
  29. tokenExpiresTime int64
  30. expiresIn int64
  31. }
  32. func NewWxApi(corpid, corpsecret string) *WxApi {
  33. return &WxApi{corpid: corpid, corpsecret: corpsecret}
  34. }
  35. // 刷新token
  36. func (a *WxApi) refreshAccessToken() error {
  37. if a.accessToken == "" || (a.tokenExpiresTime-time.Now().Unix()) < (a.expiresIn/10) {
  38. a.lock.Lock()
  39. defer a.lock.Unlock()
  40. if !((a.tokenExpiresTime - time.Now().Unix()) < (a.expiresIn / 10)) {
  41. return nil
  42. }
  43. url := GetTokenUrl
  44. url = strings.ReplaceAll(url, "ID", a.corpid)
  45. url = strings.ReplaceAll(url, "SECRET", a.corpsecret)
  46. resp, err := http.Get(url)
  47. if err != nil {
  48. return err
  49. }
  50. defer resp.Body.Close()
  51. bts, err := ioutil.ReadAll(resp.Body)
  52. if err != nil {
  53. return err
  54. }
  55. var data TokenResp
  56. err = json.Unmarshal(bts, &data)
  57. if err != nil {
  58. return err
  59. }
  60. if data.Errcode != 0 {
  61. return errors.New(fmt.Sprintf("errcode: %v,errmsg: %v", data.Errcode, data.Errmsg))
  62. }
  63. a.accessToken = data.AccessToken
  64. a.expiresIn = data.ExpiresIn
  65. a.tokenExpiresTime = time.Now().Add(time.Duration(data.ExpiresIn) * time.Second).Unix()
  66. }
  67. return nil
  68. }
  69. // GetServiceList 获取客服账号列表
  70. func (a *WxApi) GetServiceList() (err error, list []ServiceAccount) {
  71. err = a.refreshAccessToken()
  72. if err != nil {
  73. return
  74. }
  75. url := GetServiceUrl
  76. url = strings.ReplaceAll(url, "ACCESS_TOKEN", a.accessToken)
  77. res, err := http.Post(url, "application/json", strings.NewReader(`{"offset":0,"limit":999}`))
  78. if err != nil {
  79. return
  80. }
  81. defer res.Body.Close()
  82. bodyBts, err := ioutil.ReadAll(res.Body)
  83. if err != nil {
  84. return
  85. }
  86. var data ServiceListResp
  87. err = json.Unmarshal(bodyBts, &data)
  88. if err != nil {
  89. return
  90. }
  91. if data.Errcode != 0 {
  92. return errors.New(fmt.Sprintf("errcode: %v,errmsg: %v", data.Errcode, data.Errmsg)), nil
  93. }
  94. list = data.AccountList
  95. return
  96. }
  97. // GetServicerList 获取接待账号列表
  98. func (a *WxApi) GetServicerList(openKfid string) (err error, list []ServicerAccount) {
  99. err = a.refreshAccessToken()
  100. if err != nil {
  101. return
  102. }
  103. url := GetServicerUrl
  104. url = strings.ReplaceAll(url, "ACCESS_TOKEN", a.accessToken)
  105. url = strings.ReplaceAll(url, "OPEN_KFID", openKfid)
  106. res, err := http.Get(url)
  107. if err != nil {
  108. return
  109. }
  110. defer res.Body.Close()
  111. bodyBts, err := ioutil.ReadAll(res.Body)
  112. if err != nil {
  113. return
  114. }
  115. var data ServicerListResp
  116. err = json.Unmarshal(bodyBts, &data)
  117. if err != nil {
  118. return
  119. }
  120. if data.Errcode != 0 {
  121. return errors.New(fmt.Sprintf("errcode: %v,errmsg: %v", data.Errcode, data.Errmsg)), nil
  122. }
  123. list = data.ServicerList
  124. return
  125. }
  126. // 读取成员
  127. func (a *WxApi) GetStaffList(userid string) (err error, d Staff) {
  128. err = a.refreshAccessToken()
  129. if err != nil {
  130. return
  131. }
  132. url := GetStaffUrl
  133. url = strings.ReplaceAll(url, "ACCESS_TOKEN", a.accessToken)
  134. url = strings.ReplaceAll(url, "USERID", userid)
  135. res, err := http.Get(url)
  136. if err != nil {
  137. return
  138. }
  139. defer res.Body.Close()
  140. bodyBts, err := ioutil.ReadAll(res.Body)
  141. if err != nil {
  142. return
  143. }
  144. var data StaffResp
  145. err = json.Unmarshal(bodyBts, &data)
  146. if err != nil {
  147. return
  148. }
  149. if data.Errcode != 0 {
  150. return errors.New(fmt.Sprintf("errcode: %v,errmsg: %v", data.Errcode, data.Errmsg)), Staff{}
  151. }
  152. d = data.Staff
  153. return
  154. }
  155. // GetCustomerList 获取客户基础信息列表
  156. func (a *WxApi) GetCustomerList(userIds []string) (err error, list []Customer) {
  157. err = a.refreshAccessToken()
  158. if err != nil {
  159. return
  160. }
  161. url := GetCustomerUrl
  162. url = strings.ReplaceAll(url, "ACCESS_TOKEN", a.accessToken)
  163. body := CustomerParam{
  164. ExternalUseridList: userIds,
  165. NeedEnterSessionContext: 0,
  166. }
  167. bts, _ := json.Marshal(body)
  168. res, err := http.Post(url, "application/json", bytes.NewReader(bts))
  169. if err != nil {
  170. return
  171. }
  172. defer res.Body.Close()
  173. bodyBts, err := ioutil.ReadAll(res.Body)
  174. if err != nil {
  175. return
  176. }
  177. var data CustomerListResp
  178. err = json.Unmarshal(bodyBts, &data)
  179. if err != nil {
  180. return
  181. }
  182. if data.Errcode != 0 {
  183. return errors.New(fmt.Sprintf("errcode: %v,errmsg: %v", data.Errcode, data.Errmsg)), nil
  184. }
  185. list = data.CustomerList
  186. return
  187. }
  188. // GetMsgList 读取消息
  189. func (a *WxApi) GetMsgList(cursor, token, openKfid string) (err error, hasMore int, nextCursor string, list []Msg) {
  190. err = a.refreshAccessToken()
  191. if err != nil {
  192. return
  193. }
  194. url := GetMsgUrl
  195. url = strings.ReplaceAll(url, "ACCESS_TOKEN", a.accessToken)
  196. body := MsgParam{
  197. Cursor: cursor,
  198. Token: token,
  199. Limit: 1000,
  200. VoiceFormat: 0,
  201. OpenKfid: openKfid,
  202. }
  203. bts, _ := json.Marshal(body)
  204. res, err := http.Post(url, "application/json", bytes.NewReader(bts))
  205. if err != nil {
  206. return
  207. }
  208. defer res.Body.Close()
  209. bodyBts, err := ioutil.ReadAll(res.Body)
  210. if err != nil {
  211. return
  212. }
  213. var data MsgListResp
  214. err = json.Unmarshal(bodyBts, &data)
  215. if err != nil {
  216. return
  217. }
  218. if data.Errcode != 0 {
  219. err = errors.New(fmt.Sprintf("errcode: %v,errmsg: %v", data.Errcode, data.Errmsg))
  220. return
  221. }
  222. hasMore = data.HasMore
  223. nextCursor = data.NextCursor
  224. for _, msg := range data.MsgList {
  225. var m Msg
  226. m.Msgid, _ = msg["msgid"].(string)
  227. m.OpenKfid, _ = msg["open_kfid"].(string)
  228. m.ExternalUserid, _ = msg["external_userid"].(string)
  229. m.SendTime, _ = msg["send_time"].(float64)
  230. m.Origin, _ = msg["origin"].(float64)
  231. m.ServicerUserid, _ = msg["servicer_userid"].(string)
  232. m.Msgtype, _ = msg["msgtype"].(string)
  233. if m.Msgtype != "" {
  234. content, _ := msg[m.Msgtype]
  235. eventBts, _ := json.Marshal(content)
  236. m.DataInfo = string(eventBts)
  237. if m.Msgtype == "event" {
  238. var event = make(map[string]interface{})
  239. _ = json.Unmarshal(eventBts, &event)
  240. if _, b := event["event_type"].(string); b {
  241. m.OpenKfid, _ = event["open_kfid"].(string)
  242. m.ExternalUserid, _ = event["external_userid"].(string)
  243. }
  244. }
  245. }
  246. list = append(list, m)
  247. }
  248. return
  249. }
  250. // UpdateSessionState 变更会话状态
  251. func (a *WxApi) UpdateSessionState(openKfid, externalUserid, servicerUserid string, serviceState int) (err error) {
  252. err = a.refreshAccessToken()
  253. if err != nil {
  254. return
  255. }
  256. url := UpdateSessionStateUrl
  257. url = strings.ReplaceAll(url, "ACCESS_TOKEN", a.accessToken)
  258. body := UpdateSessionStateParam{
  259. OpenKfid: openKfid,
  260. ExternalUserid: externalUserid,
  261. ServiceState: serviceState,
  262. ServicerUserid: servicerUserid,
  263. }
  264. bts, _ := json.Marshal(body)
  265. res, err := http.Post(url, "application/json", bytes.NewReader(bts))
  266. if err != nil {
  267. return
  268. }
  269. defer res.Body.Close()
  270. bodyBts, err := ioutil.ReadAll(res.Body)
  271. if err != nil {
  272. return
  273. }
  274. var data UpdateSessionStateResp
  275. err = json.Unmarshal(bodyBts, &data)
  276. if err != nil {
  277. return
  278. }
  279. if data.Errcode != 0 {
  280. return errors.New(fmt.Sprintf("errcode: %v,errmsg: %v", data.Errcode, data.Errmsg))
  281. }
  282. return
  283. }