callback.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596
  1. package kf
  2. import (
  3. "encoding/xml"
  4. "github.com/silenceper/wechat/v2/util"
  5. )
  6. // SignatureOptions 微信服务器验证参数
  7. type SignatureOptions struct {
  8. Signature string `form:"msg_signature"`
  9. TimeStamp string `form:"timestamp"`
  10. Nonce string `form:"nonce"`
  11. EchoStr string `form:"echostr"`
  12. }
  13. // VerifyURL 验证请求参数是否合法并返回解密后的消息内容
  14. // //Gin框架的使用示例
  15. // r.GET("/v1/event/callback", func(c *gin.Context) {
  16. // options := kf.SignatureOptions{}
  17. // //获取回调的的校验参数
  18. // if = c.ShouldBindQuery(&options); err != nil {
  19. // c.String(http.StatusUnauthorized, "参数解析失败")
  20. // }
  21. // // 调用VerifyURL方法校验当前请求,如果合法则把解密后的内容作为响应返回给微信服务器
  22. // echo, err := kfClient.VerifyURL(options)
  23. // if err == nil {
  24. // c.String(http.StatusOK, echo)
  25. // } else {
  26. // c.String(http.StatusUnauthorized, "非法请求来源")
  27. // }
  28. // })
  29. func (r *Client) VerifyURL(options SignatureOptions) (string, error) {
  30. if options.Signature != util.Signature(r.ctx.Token, options.TimeStamp, options.Nonce, options.EchoStr) {
  31. return "", NewSDKErr(40015)
  32. }
  33. _, bData, err := util.DecryptMsg(r.corpID, options.EchoStr, r.encodingAESKey)
  34. if err != nil {
  35. return "", NewSDKErr(40016)
  36. }
  37. return string(bData), nil
  38. }
  39. // 原始回调消息内容
  40. type callbackOriginMessage struct {
  41. ToUserName string // 企业微信的CorpID,当为第三方套件回调事件时,CorpID的内容为suiteid
  42. AgentID string // 接收的应用id,可在应用的设置页面获取
  43. Encrypt string // 消息结构体加密后的字符串
  44. }
  45. // CallbackMessage 微信客服回调消息
  46. type CallbackMessage struct {
  47. ToUserName string `json:"to_user_name"` // 微信客服组件ID
  48. CreateTime int `json:"create_time"` // 消息创建时间,unix时间戳
  49. MsgType string `json:"msgtype"` // 消息的类型,此时固定为 event
  50. Event string `json:"event"` // 事件的类型,此时固定为 kf_msg_or_event
  51. Token string `json:"token"` // 调用拉取消息接口时,需要传此token,用于校验请求的合法性
  52. }
  53. // GetCallbackMessage 获取回调事件中的消息内容
  54. // //Gin框架的使用示例
  55. // r.POST("/v1/event/callback", func(c *gin.Context) {
  56. // var (
  57. // message kf.CallbackMessage
  58. // body []byte
  59. // )
  60. // // 读取原始消息内容
  61. // body, err = c.GetRawData()
  62. // if err != nil {
  63. // c.String(http.StatusInternalServerError, err.Error())
  64. // return
  65. // }
  66. // // 解析原始数据
  67. // message, err = kfClient.GetCallbackMessage(body)
  68. // if err != nil {
  69. // c.String(http.StatusInternalServerError, "消息获取失败")
  70. // return
  71. // }
  72. // fmt.Println(message)
  73. // c.String(200, "ok")
  74. // })
  75. func (r *Client) GetCallbackMessage(encryptedMsg []byte) (msg CallbackMessage, err error) {
  76. var origin callbackOriginMessage
  77. if err = xml.Unmarshal(encryptedMsg, &origin); err != nil {
  78. return msg, err
  79. }
  80. _, bData, err := util.DecryptMsg(r.corpID, origin.Encrypt, r.encodingAESKey)
  81. if err != nil {
  82. return msg, NewSDKErr(40016)
  83. }
  84. if err = xml.Unmarshal(bData, &msg); err != nil {
  85. return msg, err
  86. }
  87. return msg, err
  88. }