syncmsg.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. package kf
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "github.com/silenceper/wechat/v2/util"
  7. "github.com/silenceper/wechat/v2/work/kf/syncmsg"
  8. )
  9. const (
  10. //获取消息
  11. syncMsgAddr = "https://qyapi.weixin.qq.com/cgi-bin/kf/sync_msg?access_token=%s"
  12. )
  13. // SyncMsgOptions 获取消息查询参数
  14. type SyncMsgOptions struct {
  15. Cursor string `json:"cursor"` // 上一次调用时返回的next_cursor,第一次拉取可以不填, 不多于64字节
  16. Token string `json:"token"` // 回调事件返回的token字段,10分钟内有效;可不填,如果不填接口有严格的频率限制, 不多于128字节
  17. Limit uint `json:"limit"` // 期望请求的数据量,默认值和最大值都为1000, 注意:可能会出现返回条数少于limit的情况,需结合返回的has_more字段判断是否继续请求。
  18. }
  19. // SyncMsgSchema 获取消息查询响应内容
  20. type syncMsgSchema struct {
  21. ErrCode int32 `json:"errcode"` // 返回码
  22. ErrMsg string `json:"errmsg"` // 错误码描述
  23. NextCursor string `json:"next_cursor"` // 下次调用带上该值,则从当前的位置继续往后拉,以实现增量拉取。强烈建议对改该字段入库保存,每次请求读取带上,请求结束后更新。避免因意外丢,导致必须从头开始拉取,引起消息延迟。
  24. HasMore uint32 `json:"has_more"` // 是否还有更多数据。0-否;1-是。不能通过判断msg_list是否空来停止拉取,可能会出现has_more为1,而msg_list为空的情况
  25. MsgList []map[string]interface{} `json:"msg_list"` // 消息列表
  26. }
  27. // SyncMsgSchema 获取消息查询响应内容
  28. type SyncMsgSchema struct {
  29. ErrCode int32 `json:"errcode"` // 返回码
  30. ErrMsg string `json:"errmsg"` // 错误码描述
  31. NextCursor string `json:"next_cursor"` // 下次调用带上该值则从该key值往后拉,用于增量拉取
  32. HasMore uint32 `json:"has_more"` // 是否还有更多数据。0-否;1-是。不能通过判断msg_list是否空来停止拉取,可能会出现has_more为1,而msg_list为空的情况
  33. MsgList []syncmsg.Message `json:"msg_list"` // 消息列表
  34. }
  35. // SyncMsg 获取消息
  36. func (r *Client) SyncMsg(options SyncMsgOptions) (info SyncMsgSchema, err error) {
  37. var (
  38. accessToken string
  39. data []byte
  40. )
  41. accessToken, err = r.ctx.GetAccessToken()
  42. if err != nil {
  43. return
  44. }
  45. data, err = util.PostJSON(fmt.Sprintf(syncMsgAddr, accessToken), options)
  46. if err != nil {
  47. return
  48. }
  49. originInfo := syncMsgSchema{}
  50. if err = json.Unmarshal(data, &originInfo); err != nil {
  51. return
  52. }
  53. if originInfo.ErrCode != 0 {
  54. return info, errors.New(originInfo.ErrMsg)
  55. }
  56. msgList := make([]syncmsg.Message, 0)
  57. if len(originInfo.MsgList) > 0 {
  58. for _, msg := range originInfo.MsgList {
  59. newMsg := syncmsg.Message{}
  60. if val, ok := msg["msgid"].(string); ok {
  61. newMsg.MsgID = val
  62. }
  63. if val, ok := msg["open_kfid"].(string); ok {
  64. newMsg.OpenKFID = val
  65. }
  66. if val, ok := msg["external_userid"].(string); ok {
  67. newMsg.ExternalUserID = val
  68. }
  69. if val, ok := msg["send_time"].(float64); ok {
  70. newMsg.SendTime = uint64(val)
  71. }
  72. if val, ok := msg["origin"].(float64); ok {
  73. newMsg.Origin = uint32(val)
  74. }
  75. if val, ok := msg["msgtype"].(string); ok {
  76. newMsg.MsgType = val
  77. }
  78. if newMsg.MsgType == "event" {
  79. if event, ok := msg["event"].(map[string]interface{}); ok {
  80. if eType, ok := event["event_type"].(string); ok {
  81. newMsg.EventType = eType
  82. }
  83. }
  84. }
  85. originData, err := json.Marshal(msg)
  86. if err != nil {
  87. return info, err
  88. }
  89. newMsg.OriginData = originData
  90. msgList = append(msgList, newMsg)
  91. }
  92. }
  93. return SyncMsgSchema{
  94. ErrCode: originInfo.ErrCode,
  95. ErrMsg: originInfo.ErrMsg,
  96. NextCursor: originInfo.NextCursor,
  97. HasMore: originInfo.HasMore,
  98. MsgList: msgList,
  99. }, nil
  100. }