wechat_send_msg.go 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. package services
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "fmt"
  6. "hongze/hongze_public_api/models"
  7. "hongze/hongze_public_api/services/alarm_msg"
  8. "hongze/hongze_public_api/utils"
  9. "io/ioutil"
  10. "net/http"
  11. "strings"
  12. "time"
  13. )
  14. type SendTemplateResponse struct {
  15. Errcode int `json:"errcode"`
  16. Errmsg string `json:"errmsg"`
  17. MsgID int `json:"msgid"`
  18. }
  19. type ClearQuotaResponse struct {
  20. Errcode int `json:"errcode"`
  21. Errmsg string `json:"errmsg"`
  22. }
  23. func SendWxTemplateMsg(sendInfo *models.SendWxTemplate) (err error) {
  24. var msg string
  25. defer func() {
  26. if err != nil {
  27. go alarm_msg.SendAlarmMsg("发送模版消息失败,Err:"+err.Error()+";msg:"+msg, 3)
  28. utils.FileLog.Info(fmt.Sprintf("发送模版消息失败,Err:%s,%s", err.Error(), msg))
  29. }
  30. if msg != "" {
  31. utils.FileLog.Info(fmt.Sprintf("发送模版消息失败,msg:%s", msg))
  32. }
  33. }()
  34. utils.FileLog.Info("services SendMsg")
  35. accessToken, err := models.GetWxAccessToken()
  36. if err != nil {
  37. msg = "GetWxAccessToken Err:" + err.Error()
  38. return
  39. }
  40. if accessToken == "" {
  41. msg = "accessToken is empty"
  42. return
  43. }
  44. sendUrl := "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=" + accessToken
  45. fmt.Println("send start")
  46. utils.FileLog.Info("send start")
  47. sendMap := make(map[string]interface{})
  48. sendData := make(map[string]interface{})
  49. var uniqueCodeStr string
  50. if sendInfo.First != "" {
  51. sendData["first"] = map[string]interface{}{"value": sendInfo.First, "color": "#173177"}
  52. uniqueCodeStr += sendInfo.First
  53. }
  54. if sendInfo.Keyword1 != "" {
  55. sendData["keyword1"] = map[string]interface{}{"value": sendInfo.Keyword1, "color": "#173177"}
  56. uniqueCodeStr += sendInfo.Keyword1
  57. }
  58. if sendInfo.Keyword2 != "" {
  59. sendData["keyword2"] = map[string]interface{}{"value": sendInfo.Keyword2, "color": "#173177"}
  60. uniqueCodeStr += sendInfo.Keyword2
  61. }
  62. if sendInfo.Keyword3 != "" {
  63. sendData["keyword3"] = map[string]interface{}{"value": sendInfo.Keyword3, "color": "#173177"}
  64. uniqueCodeStr += sendInfo.Keyword3
  65. }
  66. if sendInfo.Keyword4 != "" {
  67. sendData["keyword4"] = map[string]interface{}{"value": sendInfo.Keyword4, "color": "#173177"}
  68. uniqueCodeStr += sendInfo.Keyword4
  69. }
  70. if sendInfo.TemplateId != "" {
  71. sendMap["template_id"] = sendInfo.TemplateId
  72. uniqueCodeStr += sendInfo.TemplateId
  73. }
  74. if sendInfo.RedirectUrl != "" {
  75. if strings.Contains(sendInfo.RedirectUrl, "http") || strings.Contains(sendInfo.RedirectUrl, "https") {
  76. sendMap["url"] = sendInfo.RedirectUrl
  77. } else {
  78. sendMap["miniprogram"] = sendInfo.RedirectUrl
  79. }
  80. uniqueCodeStr += sendInfo.RedirectUrl
  81. }
  82. sendMap["data"] = sendData
  83. uniqueCode := utils.MD5(uniqueCodeStr)
  84. err = sendTemplateMsg(sendUrl, sendMap, sendInfo.OpenIdArr, sendInfo.Resource, uniqueCode, sendInfo.SendType)
  85. if err!=nil {
  86. utils.FileLog.Info("send err:"+err.Error())
  87. }
  88. fmt.Println("send end")
  89. utils.FileLog.Info("send end")
  90. return
  91. }
  92. func sendTemplateMsg(sendUrl string, sendMap map[string]interface{}, openIdArr []string, resource, uniqueCode string, sendType int) (err error) {
  93. existList, err := models.GetTemplateRecordByUniqueCode(uniqueCode)
  94. if err != nil && err.Error() != utils.ErrNoRow() {
  95. utils.FileLog.Info(fmt.Sprintf("GetTemplateRecordByUniqueCode Err:%s", err.Error()))
  96. return err
  97. }
  98. if len(existList) == 0 {
  99. for _, openId := range openIdArr {
  100. sendMap["touser"] = openId
  101. data, err := json.Marshal(sendMap)
  102. if err != nil {
  103. fmt.Println("SendTemplateMsgOne Marshal Err:", err.Error())
  104. utils.FileLog.Info(fmt.Sprintf("SendTemplateMsgOne Marshal Err:%s", err.Error()))
  105. return err
  106. }
  107. err = toSendTemplateMsg(sendUrl, data, resource, sendType, openId, uniqueCode)
  108. if err != nil {
  109. fmt.Println("send err:", err.Error())
  110. utils.FileLog.Info(fmt.Sprintf("ToSendTemplateMsg Err:%s", err.Error()))
  111. }
  112. }
  113. } else {
  114. existOpenIdMap := make(map[string]string)
  115. for _, v := range existList {
  116. existOpenIdMap[v.OpenId] = v.OpenId
  117. }
  118. for _, openId := range openIdArr {
  119. if _, ok := existOpenIdMap[openId]; !ok {
  120. sendMap["touser"] = openId
  121. data, err := json.Marshal(sendMap)
  122. if err != nil {
  123. fmt.Println("SendTemplateMsgOne Marshal Err:", err.Error())
  124. utils.FileLog.Info(fmt.Sprintf("SendTemplateMsgOne Marshal Err:%s", err.Error()))
  125. return err
  126. }
  127. err = toSendTemplateMsg(sendUrl, data, resource, sendType, openId, uniqueCode)
  128. if err != nil {
  129. fmt.Println("send err:", err.Error())
  130. utils.FileLog.Info(fmt.Sprintf("ToSendTemplateMsg Err:%s", err.Error()))
  131. }
  132. }
  133. }
  134. }
  135. return
  136. }
  137. func toSendTemplateMsg(sendUrl string, data []byte, resource string, sendType int, openId, uniqueCode string) (err error) {
  138. var msg string
  139. utils.FileLog.Info("Send:" + string(data))
  140. client := http.Client{}
  141. resp, err := client.Post(sendUrl, "application/json", bytes.NewBuffer(data))
  142. if err != nil {
  143. return
  144. }
  145. defer resp.Body.Close()
  146. body, _ := ioutil.ReadAll(resp.Body)
  147. utils.FileLog.Info("SendResult:" + string(body))
  148. var templateResponse SendTemplateResponse
  149. err = json.Unmarshal(body, &templateResponse)
  150. if err != nil {
  151. utils.FileLog.Info(fmt.Sprintf("SendResult Unmarshal Err:%s", err.Error()))
  152. return err
  153. }
  154. //新增模板消息推送记录
  155. {
  156. tr := new(models.UserTemplateRecord)
  157. tr.OpenId = openId
  158. tr.Resource = resource
  159. tr.SendData = string(data)
  160. tr.Result = string(body)
  161. tr.CreateDate = time.Now().Format(utils.FormatDate)
  162. tr.CreateTime = time.Now().Format(utils.FormatDateTime)
  163. if templateResponse.Errcode == 0 {
  164. tr.SendStatus = 1
  165. } else {
  166. tr.SendStatus = 0
  167. }
  168. tr.UniqueCode = uniqueCode
  169. tr.SendType = sendType
  170. go func() {
  171. err = models.AddUserTemplateRecord(tr)
  172. if err != nil {
  173. utils.FileLog.Info(fmt.Sprintf("AddUserTemplateRecord Err:%s", err.Error()))
  174. }
  175. }()
  176. }
  177. accessToken, err := models.GetWxAccessToken()
  178. if err != nil {
  179. msg = "GetWxAccessToken Err:" + err.Error()
  180. utils.FileLog.Info("获取Token失败,msg:" + msg)
  181. return
  182. }
  183. if accessToken == "" {
  184. msg = "accessToken is empty"
  185. utils.FileLog.Info("accessToken为空,msg:" + msg)
  186. return
  187. }
  188. //模板消息发送超过当日10万次限制错误处理
  189. if templateResponse.Errcode == 45009 {
  190. key := "CACHE_SendTemplateMsg_ERR"
  191. isExist := utils.Rc.IsExist(key)
  192. if isExist == true {
  193. return
  194. } else {
  195. result, _ := json.Marshal(templateResponse)
  196. if err != nil {
  197. utils.FileLog.Info(fmt.Sprintf("templateResponse Marshal Err:%s", err.Error()))
  198. return err
  199. }
  200. //发送邮件提醒异常
  201. go alarm_msg.SendAlarmMsg("模板消息发送超过当日10万次限制,templateResponse = "+string(result), 3)
  202. //go utils.SendEmail("异常提醒:", "模板消息发送超过当日10万次限制,templateResponse = "+string(result), utils.EmailSendToUsers)
  203. //设置3分钟缓存,不允许重复添加
  204. utils.Rc.SetNX(key, 1, 6*time.Minute)
  205. }
  206. //清空发送次数
  207. sendUrl := fmt.Sprintf("https://api.weixin.qq.com/cgi-bin/clear_quota?access_token=%s", accessToken)
  208. clearData := make(map[string]interface{})
  209. clearData["appid"] = utils.WxAppId
  210. clearJson, _ := json.Marshal(clearData)
  211. utils.FileLog.Info("clear_quota data:" + string(clearJson))
  212. resp, err := client.Post(sendUrl, "application/json", bytes.NewBuffer(clearJson))
  213. if err != nil {
  214. return err
  215. }
  216. defer resp.Body.Close()
  217. clearBody, err := ioutil.ReadAll(resp.Body)
  218. utils.FileLog.Info("clear_quota result:" + string(clearBody))
  219. var clearQuotaResponse ClearQuotaResponse
  220. err = json.Unmarshal(clearBody, &clearQuotaResponse)
  221. if err != nil {
  222. utils.FileLog.Info(fmt.Sprintf("clearQuotaResponse Unmarshal Err:%s", err.Error()))
  223. return err
  224. }
  225. if clearQuotaResponse.Errcode == 0 {
  226. //发送邮件解决异常
  227. go alarm_msg.SendAlarmMsg("异常已解决,自动清理限制接口,调用成功", 3)
  228. //go utils.SendEmail("异常已解决:", "自动清理限制接口,调用成功", utils.EmailSendToUsers)
  229. //重新推送一次
  230. toSendTemplateMsg(sendUrl, data, resource, sendType, openId, uniqueCode)
  231. }
  232. }
  233. return
  234. }