template_msg.go 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. package wechat
  2. import (
  3. "bytes"
  4. "context"
  5. "encoding/json"
  6. "errors"
  7. "fmt"
  8. "hongze/hongze_yb/global"
  9. "hongze/hongze_yb/models/tables/user_record"
  10. "hongze/hongze_yb/models/tables/user_template_record"
  11. "hongze/hongze_yb/services/alarm_msg"
  12. "hongze/hongze_yb/utils"
  13. "io/ioutil"
  14. "net/http"
  15. "time"
  16. )
  17. var (
  18. TemplateMsgSendUrl = "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=%s"
  19. TemplateMsgClearQuotaUrl = "https://api.weixin.qq.com/cgi-bin/clear_quota?access_token=%s"
  20. )
  21. type TemplateMsgSendClient struct {
  22. AccessToken string
  23. Data []byte
  24. }
  25. type SendTemplateResponse struct {
  26. Errcode int `json:"errcode"`
  27. Errmsg string `json:"errmsg"`
  28. MsgID int `json:"msgid"`
  29. }
  30. type ClearQuotaResponse struct {
  31. Errcode int `json:"errcode"`
  32. Errmsg string `json:"errmsg"`
  33. }
  34. type OpenIdList struct {
  35. OpenId string
  36. UserId int
  37. }
  38. // TemplateMsgSendClient.SendMsg 推送消息
  39. func (c *TemplateMsgSendClient) SendMsg() (sendRes *SendTemplateResponse, err error) {
  40. // 请求接口
  41. sendUrl := fmt.Sprintf(TemplateMsgSendUrl, c.AccessToken)
  42. client := http.Client{}
  43. resp, err := client.Post(sendUrl, "application/json", bytes.NewBuffer(c.Data))
  44. if err != nil {
  45. return
  46. }
  47. defer func() {
  48. _ = resp.Body.Close()
  49. }()
  50. body, _ := ioutil.ReadAll(resp.Body)
  51. if err = json.Unmarshal(body, &sendRes); err != nil {
  52. return
  53. }
  54. // 模板消息发送超过当日10万次限制错误处理
  55. if sendRes.Errcode == 45009 {
  56. // 发送提示邮件
  57. go alarm_msg.SendAlarmMsg("模板消息发送超过当日10万次限制, SendTemplateResponse: "+string(body), 3)
  58. // 清理限制
  59. clearRes, e := c.ClearQuota()
  60. if e != nil {
  61. err = e
  62. return
  63. }
  64. if clearRes.Errcode != 0 {
  65. clearJson, _ := json.Marshal(clearRes)
  66. go alarm_msg.SendAlarmMsg("自动清理模板消息限制接口, 调用失败, ClearQuotaResponse: "+string(clearJson), 3)
  67. return
  68. }
  69. // 发送成功邮件
  70. go alarm_msg.SendAlarmMsg("自动清理模板消息限制接口, 调用成功", 1)
  71. // 重新推送
  72. go func() {
  73. reSend, e := c.SendMsg()
  74. if e != nil {
  75. reSendJson, _ := json.Marshal(reSend)
  76. alarm_msg.SendAlarmMsg("重新推送模板消息失败, SendTemplateResponse: "+string(reSendJson), 3)
  77. }
  78. }()
  79. }
  80. if sendRes.Errcode != 0 {
  81. err = errors.New("推送模板消息失败, SendTemplateResponse: " + string(body))
  82. }
  83. return
  84. }
  85. // TemplateMsgSendClient.ClearQuota 清除发送超过当日10万次限制
  86. func (c *TemplateMsgSendClient) ClearQuota() (result *ClearQuotaResponse, err error) {
  87. key := "CACHE_SendTemplateMsg_ERR"
  88. exists, _ := global.Redis.Exists(context.TODO(), key).Result()
  89. if exists == 1 {
  90. return
  91. }
  92. _ = global.Redis.SetEX(context.TODO(), key, 1, 6*time.Minute)
  93. sendUrl := fmt.Sprintf(TemplateMsgClearQuotaUrl, c.AccessToken)
  94. client := http.Client{}
  95. clearData := make(map[string]interface{})
  96. clearData["appid"] = WxAppId
  97. clearJson, _ := json.Marshal(clearData)
  98. resp, err := client.Post(sendUrl, "application/json", bytes.NewBuffer(clearJson))
  99. if err != nil {
  100. return
  101. }
  102. defer func() {
  103. _ = resp.Body.Close()
  104. }()
  105. clearBody, err := ioutil.ReadAll(resp.Body)
  106. err = json.Unmarshal(clearBody, &result)
  107. return
  108. }
  109. // AddUserTemplateRecord 新增模板消息推送记录
  110. func AddUserTemplateRecord(userId, sendStatus, sendType int, openid, resource, sendData, result string) (err error) {
  111. item := &user_template_record.UserTemplateRecord{
  112. UserID: userId,
  113. OpenID: openid,
  114. Resource: resource,
  115. SendData: sendData,
  116. Result: result,
  117. CreateDate: time.Now().Format(utils.FormatDate),
  118. CreateTime: time.Now().Format(utils.FormatDateTime),
  119. SendStatus: sendStatus,
  120. SendType: sendType,
  121. }
  122. err = item.Create()
  123. return
  124. }
  125. // SendMultiTemplateMsg 推送模板消息至多个用户
  126. func SendMultiTemplateMsg(sendMap map[string]interface{}, items []*OpenIdList, resource string, sendType int) (err error) {
  127. ws := GetWxChat()
  128. accessToken, err := ws.GetAccessToken()
  129. if err != nil {
  130. return
  131. }
  132. for _, item := range items {
  133. sendMap["touser"] = item.OpenId
  134. data, e := json.Marshal(sendMap)
  135. if e != nil {
  136. err = e
  137. return
  138. }
  139. ts := &TemplateMsgSendClient{
  140. AccessToken: accessToken,
  141. Data: data,
  142. }
  143. result, e := ts.SendMsg()
  144. if result == nil {
  145. return
  146. }
  147. // 推送消息记录
  148. {
  149. go func(v *OpenIdList) {
  150. sendStatus := 1
  151. if e != nil {
  152. sendStatus = 0
  153. }
  154. resultJson, _ := json.Marshal(result)
  155. _ = AddUserTemplateRecord(v.UserId, sendStatus, sendType, v.OpenId, resource, string(data), string(resultJson))
  156. }(item)
  157. }
  158. if e != nil {
  159. err = e
  160. return
  161. }
  162. }
  163. return
  164. }
  165. // SendQuestionReplyWxMsg 推送研报小程序模板消息-问答社区回复
  166. func SendQuestionReplyWxMsg(questionId, userId int, questionTitle string) (err error) {
  167. if userId == 0 {
  168. return
  169. }
  170. var errMsg string
  171. defer func() {
  172. if err != nil {
  173. alarmMsg := fmt.Sprintf("SendQuestionReplyWxMsg-推送问答社区回复模版消息失败; QuestionId: %d; Err: %s; Msg: %s", questionId, err.Error(), errMsg)
  174. go alarm_msg.SendAlarmMsg(alarmMsg, 3)
  175. }
  176. }()
  177. // 校验用户是否取消关注公众号
  178. userRecord, e := user_record.GetByUserId(userId, utils.USER_RECORD_PLATFORM_RDDP)
  179. if e != nil {
  180. // 用户无公众号信息
  181. if e == utils.ErrNoRow {
  182. return
  183. }
  184. err = errors.New("获取用户Record信息失败, Err:" + e.Error())
  185. return
  186. }
  187. // 取消关注则不推送
  188. if userRecord.Subscribe == 0 {
  189. return
  190. }
  191. openIdList := make([]*OpenIdList, 0)
  192. openIdList = append(openIdList, &OpenIdList{
  193. OpenId: userRecord.OpenID,
  194. UserId: userId,
  195. })
  196. sendMap := make(map[string]interface{})
  197. sendData := make(map[string]interface{})
  198. first := "您好,您的提问已被回复"
  199. keyword1 := questionTitle
  200. keyword2 := "待查看"
  201. remark := "请点击详情查看回复"
  202. sendData["first"] = map[string]interface{}{"value": first, "color": "#173177"}
  203. sendData["keyword1"] = map[string]interface{}{"value": keyword1, "color": "#173177"}
  204. sendData["keyword2"] = map[string]interface{}{"value": keyword2, "color": "#173177"}
  205. sendData["remark"] = map[string]interface{}{"value": remark, "color": "#173177"}
  206. sendMap["template_id"] = TemplateIdWithCommunityQuestion
  207. sendMap["data"] = sendData
  208. wxAppPath := fmt.Sprintf("pages-question/answerDetail?id=%d", questionId)
  209. if global.CONFIG.Serve.RunMode == "debug" {
  210. // 仅测试环境测试用
  211. wxAppPath = "pages-report/reportDetail?reportId=3800"
  212. }
  213. if wxAppPath != "" {
  214. sendMap["miniprogram"] = map[string]interface{}{"appid": WxYbAppId, "pagepath": wxAppPath}
  215. }
  216. err = SendMultiTemplateMsg(sendMap, openIdList, wxAppPath, utils.TEMPLATE_MSG_YB_COMMUNITY_QUESTION)
  217. return
  218. }