wechat_send_msg.go 8.5 KB

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