wechat_send_msg.go 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  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. fmt.Println("send start")
  38. utils.FileLog.Info("send start")
  39. sendMap := make(map[string]interface{})
  40. sendData := make(map[string]interface{})
  41. var uniqueCodeStr string
  42. if sendInfo.First != "" {
  43. sendData["first"] = map[string]interface{}{"value": sendInfo.First, "color": "#173177"}
  44. uniqueCodeStr += sendInfo.First
  45. }
  46. if sendInfo.Keyword1 != "" {
  47. sendData["keyword1"] = map[string]interface{}{"value": sendInfo.Keyword1, "color": "#173177"}
  48. uniqueCodeStr += sendInfo.Keyword1
  49. }
  50. if sendInfo.Keyword2 != "" {
  51. sendData["keyword2"] = map[string]interface{}{"value": sendInfo.Keyword2, "color": "#173177"}
  52. uniqueCodeStr += sendInfo.Keyword2
  53. }
  54. if sendInfo.Keyword3 != "" {
  55. sendData["keyword3"] = map[string]interface{}{"value": sendInfo.Keyword3, "color": "#173177"}
  56. uniqueCodeStr += sendInfo.Keyword3
  57. }
  58. if sendInfo.Keyword4 != "" {
  59. sendData["keyword4"] = map[string]interface{}{"value": sendInfo.Keyword4, "color": "#173177"}
  60. uniqueCodeStr += sendInfo.Keyword4
  61. }
  62. if sendInfo.TemplateId != "" {
  63. sendMap["template_id"] = sendInfo.TemplateId
  64. uniqueCodeStr += sendInfo.TemplateId
  65. }
  66. if sendInfo.RedirectUrl != "" {
  67. if strings.Contains(sendInfo.RedirectUrl, "http") || strings.Contains(sendInfo.RedirectUrl, "https") || sendInfo.RedirectTarget == 0 {
  68. sendMap["url"] = sendInfo.RedirectUrl
  69. } else {
  70. var xcxAppId string
  71. if sendInfo.RedirectTarget == 1 {
  72. xcxAppId = utils.WxYbAppId
  73. } else if sendInfo.RedirectTarget == 2 {
  74. xcxAppId = utils.WxCrmAppId
  75. } else if sendInfo.RedirectTarget == 3 {
  76. xcxAppId = utils.WxCygxAppId
  77. } else {
  78. err = errors.New("无效的微信小程序跳转方式:RedirectTarget" + strconv.Itoa(sendInfo.RedirectTarget))
  79. }
  80. sendMap["miniprogram"] = map[string]interface{}{"appid": xcxAppId, "pagepath": sendInfo.RedirectUrl}
  81. }
  82. uniqueCodeStr += sendInfo.RedirectUrl
  83. }
  84. sendMap["data"] = sendData
  85. uniqueCode := utils.MD5(uniqueCodeStr)
  86. err = sendTemplateMsg(sendMap, sendInfo.OpenIdArr, sendInfo.Resource, uniqueCode, sendInfo.SendType)
  87. if err != nil {
  88. utils.FileLog.Info("send err:" + err.Error())
  89. }
  90. fmt.Println("send end")
  91. utils.FileLog.Info("send end")
  92. return
  93. }
  94. // sendTemplateMsg 整理openid以及以往历史推送记录,移除已经推送的记录,并开始依次推送
  95. func sendTemplateMsg(sendMap map[string]interface{}, openIdArr []string, resource, uniqueCode string, sendType int) (err error) {
  96. existList, err := models.GetTemplateRecordByUniqueCode(uniqueCode)
  97. if err != nil && err.Error() != utils.ErrNoRow() {
  98. utils.FileLog.Info(fmt.Sprintf("GetTemplateRecordByUniqueCode Err:%s", err.Error()))
  99. return err
  100. }
  101. if len(existList) == 0 {
  102. for _, openId := range openIdArr {
  103. sendMap["touser"] = openId
  104. data, err := json.Marshal(sendMap)
  105. if err != nil {
  106. fmt.Println("SendTemplateMsgOne Marshal Err:", err.Error())
  107. utils.FileLog.Info(fmt.Sprintf("SendTemplateMsgOne Marshal Err:%s", err.Error()))
  108. return err
  109. }
  110. err = toSendTemplateMsg(data, resource, sendType, openId, uniqueCode)
  111. if err != nil {
  112. fmt.Println("send err:", err.Error())
  113. utils.FileLog.Info(fmt.Sprintf("ToSendTemplateMsg Err:%s", err.Error()))
  114. }
  115. }
  116. } else {
  117. existOpenIdMap := make(map[string]string)
  118. for _, v := range existList {
  119. existOpenIdMap[v.OpenId] = v.OpenId
  120. }
  121. for _, openId := range openIdArr {
  122. if _, ok := existOpenIdMap[openId]; !ok {
  123. sendMap["touser"] = openId
  124. data, err := json.Marshal(sendMap)
  125. if err != nil {
  126. fmt.Println("SendTemplateMsgOne Marshal Err:", err.Error())
  127. utils.FileLog.Info(fmt.Sprintf("SendTemplateMsgOne Marshal Err:%s", err.Error()))
  128. return err
  129. }
  130. err = toSendTemplateMsg( data, resource, sendType, openId, uniqueCode)
  131. if err != nil {
  132. fmt.Println("send err:", err.Error())
  133. utils.FileLog.Info(fmt.Sprintf("ToSendTemplateMsg Err:%s", err.Error()))
  134. }
  135. }
  136. }
  137. }
  138. return
  139. }
  140. // toSendTemplateMsg 实际推送微信
  141. func toSendTemplateMsg(data []byte, resource string, sendType int, openId, uniqueCode string) (err error) {
  142. utils.FileLog.Info("Send:" + string(data))
  143. //获取accessToken
  144. accessToken, err,errMsg := getWxAccessToken()
  145. if err != nil {
  146. utils.FileLog.Info(fmt.Sprintf("获取Token失败,err:%s,errMsg:%s",err.Error(), errMsg))
  147. return
  148. }
  149. sendUrl := "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=" + accessToken
  150. client := http.Client{}
  151. resp, err := client.Post(sendUrl, "application/json", bytes.NewBuffer(data))
  152. if err != nil {
  153. return
  154. }
  155. defer resp.Body.Close()
  156. body, _ := ioutil.ReadAll(resp.Body)
  157. utils.FileLog.Info("SendResult:" + string(body))
  158. var templateResponse SendTemplateResponse
  159. err = json.Unmarshal(body, &templateResponse)
  160. if err != nil {
  161. utils.FileLog.Info(fmt.Sprintf("SendResult Unmarshal Err:%s", err.Error()))
  162. return err
  163. }
  164. //新增模板消息推送记录
  165. {
  166. tr := new(models.UserTemplateRecord)
  167. tr.OpenId = openId
  168. tr.Resource = resource
  169. tr.SendData = string(data)
  170. tr.Result = string(body)
  171. tr.CreateDate = time.Now().Format(utils.FormatDate)
  172. tr.CreateTime = time.Now().Format(utils.FormatDateTime)
  173. if templateResponse.Errcode == 0 {
  174. tr.SendStatus = 1
  175. } else {
  176. tr.SendStatus = 0
  177. }
  178. tr.UniqueCode = uniqueCode
  179. tr.SendType = sendType
  180. go func() {
  181. err = models.AddUserTemplateRecord(tr)
  182. if err != nil {
  183. utils.FileLog.Info(fmt.Sprintf("AddUserTemplateRecord Err:%s", err.Error()))
  184. }
  185. }()
  186. }
  187. //accessToken过期
  188. if templateResponse.Errcode == 40001{
  189. //强刷token并重新推送
  190. accessToken,err,errMsg = refreshWxAccessToken()
  191. if err != nil{
  192. utils.FileLog.Info(fmt.Sprintf("refreshWxAccessToken Err:%s", err.Error()))
  193. return err
  194. }
  195. return toSendTemplateMsg(data, resource, sendType, openId, uniqueCode)
  196. }
  197. //模板消息发送超过当日10万次限制错误处理
  198. if templateResponse.Errcode == 45009 {
  199. key := "CACHE_SendTemplateMsg_ERR"
  200. isExist := utils.Rc.IsExist(key)
  201. if isExist == true {
  202. return
  203. } else {
  204. result, _ := json.Marshal(templateResponse)
  205. if err != nil {
  206. utils.FileLog.Info(fmt.Sprintf("templateResponse Marshal Err:%s", err.Error()))
  207. return err
  208. }
  209. //发送邮件提醒异常
  210. go alarm_msg.SendAlarmMsg("模板消息发送超过当日10万次限制,templateResponse = "+string(result), 3)
  211. //go utils.SendEmail("异常提醒:", "模板消息发送超过当日10万次限制,templateResponse = "+string(result), utils.EmailSendToUsers)
  212. //设置3分钟缓存,不允许重复添加
  213. utils.Rc.SetNX(key, 1, 6*time.Minute)
  214. }
  215. //清空发送次数
  216. sendUrl := fmt.Sprintf("https://api.weixin.qq.com/cgi-bin/clear_quota?access_token=%s", accessToken)
  217. clearData := make(map[string]interface{})
  218. clearData["appid"] = utils.WxAppId
  219. clearJson, _ := json.Marshal(clearData)
  220. utils.FileLog.Info("clear_quota data:" + string(clearJson))
  221. resp, err := client.Post(sendUrl, "application/json", bytes.NewBuffer(clearJson))
  222. if err != nil {
  223. return err
  224. }
  225. defer resp.Body.Close()
  226. clearBody, err := ioutil.ReadAll(resp.Body)
  227. utils.FileLog.Info("clear_quota result:" + string(clearBody))
  228. var clearQuotaResponse ClearQuotaResponse
  229. err = json.Unmarshal(clearBody, &clearQuotaResponse)
  230. if err != nil {
  231. utils.FileLog.Info(fmt.Sprintf("clearQuotaResponse Unmarshal Err:%s", err.Error()))
  232. return err
  233. }
  234. if clearQuotaResponse.Errcode == 0 {
  235. //发送邮件解决异常
  236. go alarm_msg.SendAlarmMsg("异常已解决,自动清理限制接口,调用成功", 3)
  237. //go utils.SendEmail("异常已解决:", "自动清理限制接口,调用成功", utils.EmailSendToUsers)
  238. //重新推送一次
  239. toSendTemplateMsg(data, resource, sendType, openId, uniqueCode)
  240. }
  241. }
  242. return
  243. }
  244. // getWxAccessToken 获取微信token
  245. func getWxAccessToken()(accessToken string,err error,errMsg string){
  246. accessToken, err = utils.Rc.RedisString(utils.CACHE_WX_ACCESS_TOKEN_HZ)
  247. if err != nil {
  248. errMsg = "GetWxAccessToken Err:" + err.Error()
  249. utils.FileLog.Info("获取Token失败,msg:" + errMsg)
  250. return
  251. }
  252. //取到数据后就直接返回了,没有后续了
  253. if accessToken != ""{
  254. return
  255. }
  256. //缓存中没有取到数据,那么需要去强制刷新新的accessToken
  257. return refreshWxAccessToken()
  258. }
  259. // refreshWxAccessToken 强制刷新微信token
  260. func refreshWxAccessToken()(accessToken string,err error,errMsg string){
  261. //调用微信官方接口获取新的accessToken
  262. wxAccessToken, tmpErr := models.GetWxToken()
  263. if tmpErr != nil {
  264. errMsg = "通过微信接口获取accessToken失败 Err:" + err.Error()
  265. err = tmpErr
  266. return
  267. }
  268. //如果没有token数据
  269. if wxAccessToken.AccessToken == "" {
  270. errMsg = "微信返回的accessToken异常: Err:" + wxAccessToken.Errmsg
  271. err = errors.New(errMsg)
  272. return
  273. }
  274. accessToken = wxAccessToken.AccessToken
  275. //更新mysql的accessToken
  276. expiresIn := time.Now().Add(time.Duration(wxAccessToken.ExpiresIn) * time.Second).Unix()
  277. err = models.ModifyAccessToken(wxAccessToken.AccessToken,expiresIn)
  278. if err!=nil{
  279. errMsg = "更新mysql中的accessToken失败 Err:" + err.Error()
  280. return
  281. }
  282. //更新redis的accessToken(过期时间提前十分钟)
  283. redisTimeExpire := time.Duration(wxAccessToken.ExpiresIn - 600) * time.Second
  284. utils.Rc.SetNX(utils.CACHE_WX_ACCESS_TOKEN_HZ,accessToken,redisTimeExpire)
  285. return
  286. }