wechat_send_msg.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  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.WxAppId != "" {
  43. uniqueCodeStr += sendInfo.WxAppId
  44. }
  45. if sendInfo.First != "" {
  46. sendData["first"] = map[string]interface{}{"value": sendInfo.First, "color": "#173177"}
  47. uniqueCodeStr += sendInfo.First
  48. }
  49. if sendInfo.Keyword1 != "" {
  50. sendData["keyword1"] = map[string]interface{}{"value": sendInfo.Keyword1, "color": "#173177"}
  51. uniqueCodeStr += sendInfo.Keyword1
  52. }
  53. if sendInfo.Keyword2 != "" {
  54. sendData["keyword2"] = map[string]interface{}{"value": sendInfo.Keyword2, "color": "#173177"}
  55. uniqueCodeStr += sendInfo.Keyword2
  56. }
  57. if sendInfo.Keyword3 != "" {
  58. sendData["keyword3"] = map[string]interface{}{"value": sendInfo.Keyword3, "color": "#173177"}
  59. uniqueCodeStr += sendInfo.Keyword3
  60. }
  61. if sendInfo.Keyword4 != "" {
  62. sendData["keyword4"] = map[string]interface{}{"value": sendInfo.Keyword4, "color": "#173177"}
  63. uniqueCodeStr += sendInfo.Keyword4
  64. }
  65. if sendInfo.Keyword5 != "" {
  66. sendData["keyword5"] = map[string]interface{}{"value": sendInfo.Keyword5, "color": "#173177"}
  67. uniqueCodeStr += sendInfo.Keyword5
  68. }
  69. if sendInfo.Productname != "" {
  70. sendData["productname"] = map[string]interface{}{"value": sendInfo.Productname, "color": "#173177"}
  71. uniqueCodeStr += sendInfo.Productname
  72. }
  73. if sendInfo.Date != "" {
  74. sendData["date"] = map[string]interface{}{"value": sendInfo.Date, "color": "#173177"}
  75. uniqueCodeStr += sendInfo.Date
  76. }
  77. if sendInfo.Remark != "" {
  78. sendData["remark"] = map[string]interface{}{"value": sendInfo.Remark, "color": "#173177"}
  79. uniqueCodeStr += sendInfo.Remark
  80. }
  81. if sendInfo.TemplateId != "" {
  82. sendMap["template_id"] = sendInfo.TemplateId
  83. uniqueCodeStr += sendInfo.TemplateId
  84. }
  85. if sendInfo.RedirectUrl != "" {
  86. if strings.Contains(sendInfo.RedirectUrl, "http") || strings.Contains(sendInfo.RedirectUrl, "https") || sendInfo.RedirectTarget == 0 {
  87. sendMap["url"] = sendInfo.RedirectUrl
  88. } else {
  89. var xcxAppId string
  90. if sendInfo.RedirectTarget == 1 {
  91. xcxAppId = utils.WxYbAppId
  92. } else if sendInfo.RedirectTarget == 2 {
  93. xcxAppId = utils.WxCrmAppId
  94. } else if sendInfo.RedirectTarget == 3 {
  95. xcxAppId = utils.WxCygxAppId
  96. } else if sendInfo.RedirectTarget == 4 {
  97. xcxAppId = utils.WxCopyYbAppId
  98. } else {
  99. err = errors.New("无效的微信小程序跳转方式:RedirectTarget" + strconv.Itoa(sendInfo.RedirectTarget))
  100. return err
  101. }
  102. sendMap["miniprogram"] = map[string]interface{}{"appid": xcxAppId, "pagepath": sendInfo.RedirectUrl}
  103. }
  104. uniqueCodeStr += sendInfo.RedirectUrl
  105. }
  106. sendMap["data"] = sendData
  107. uniqueCode := utils.MD5(uniqueCodeStr)
  108. err = sendTemplateMsg(sendInfo.WxAppId, sendMap, sendInfo.OpenIdArr, sendInfo.Resource, uniqueCode, sendInfo.SendType)
  109. if err != nil {
  110. utils.FileLog.Info("send err:" + err.Error())
  111. }
  112. fmt.Println("send end")
  113. utils.FileLog.Info("send end")
  114. return
  115. }
  116. // sendTemplateMsg 整理openid以及以往历史推送记录,移除已经推送的记录,并开始依次推送
  117. func sendTemplateMsg(wxAppId string, sendMap map[string]interface{}, openIdArr []string, resource, uniqueCode string, sendType int) (err error) {
  118. existList, err := models.GetTemplateRecordByUniqueCode(uniqueCode)
  119. if err != nil && err.Error() != utils.ErrNoRow() {
  120. utils.FileLog.Info(fmt.Sprintf("GetTemplateRecordByUniqueCode Err:%s", err.Error()))
  121. return err
  122. }
  123. if len(existList) == 0 {
  124. for _, openId := range openIdArr {
  125. sendMap["touser"] = openId
  126. data, err := json.Marshal(sendMap)
  127. if err != nil {
  128. fmt.Println("SendTemplateMsgOne Marshal Err:", err.Error())
  129. utils.FileLog.Info(fmt.Sprintf("SendTemplateMsgOne Marshal Err:%s", err.Error()))
  130. return err
  131. }
  132. err = toSendTemplateMsg(wxAppId, data, resource, sendType, openId, uniqueCode)
  133. if err != nil {
  134. fmt.Println("send err:", err.Error())
  135. utils.FileLog.Info(fmt.Sprintf("ToSendTemplateMsg Err:%s", err.Error()))
  136. }
  137. }
  138. } else {
  139. existOpenIdMap := make(map[string]string)
  140. for _, v := range existList {
  141. existOpenIdMap[v.OpenId] = v.OpenId
  142. }
  143. for _, openId := range openIdArr {
  144. if _, ok := existOpenIdMap[openId]; !ok {
  145. sendMap["touser"] = openId
  146. data, err := json.Marshal(sendMap)
  147. if err != nil {
  148. fmt.Println("SendTemplateMsgOne Marshal Err:", err.Error())
  149. utils.FileLog.Info(fmt.Sprintf("SendTemplateMsgOne Marshal Err:%s", err.Error()))
  150. return err
  151. }
  152. err = toSendTemplateMsg(wxAppId, data, resource, sendType, openId, uniqueCode)
  153. if err != nil {
  154. fmt.Println("send err:", err.Error())
  155. utils.FileLog.Info(fmt.Sprintf("ToSendTemplateMsg Err:%s", err.Error()))
  156. }
  157. }
  158. }
  159. }
  160. return
  161. }
  162. // toSendTemplateMsg 实际推送微信
  163. func toSendTemplateMsg(wxAppId string, data []byte, resource string, sendType int, openId, uniqueCode string) (err error) {
  164. utils.FileLog.Info("Send:" + string(data))
  165. //获取accessToken
  166. var accessToken string
  167. var errMsg string
  168. accessToken, err, errMsg = getWxAccessToken(wxAppId)
  169. if err != nil {
  170. utils.FileLog.Info(fmt.Sprintf("获取Token失败,err:%s,errMsg:%s", err.Error(), errMsg))
  171. return
  172. }
  173. sendUrl := "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=" + accessToken
  174. client := http.Client{}
  175. resp, err := client.Post(sendUrl, "application/json", bytes.NewBuffer(data))
  176. if err != nil {
  177. return
  178. }
  179. defer resp.Body.Close()
  180. body, _ := ioutil.ReadAll(resp.Body)
  181. utils.FileLog.Info("SendResult:" + string(body))
  182. var templateResponse SendTemplateResponse
  183. err = json.Unmarshal(body, &templateResponse)
  184. if err != nil {
  185. utils.FileLog.Info(fmt.Sprintf("SendResult Unmarshal Err:%s", err.Error()))
  186. return err
  187. }
  188. //新增模板消息推送记录
  189. {
  190. tr := new(models.UserTemplateRecord)
  191. tr.OpenId = openId
  192. tr.Resource = resource
  193. tr.SendData = string(data)
  194. tr.Result = string(body)
  195. tr.CreateDate = time.Now().Format(utils.FormatDate)
  196. tr.CreateTime = time.Now().Format(utils.FormatDateTime)
  197. if templateResponse.Errcode == 0 {
  198. tr.SendStatus = 1
  199. } else {
  200. tr.SendStatus = 0
  201. }
  202. tr.UniqueCode = uniqueCode
  203. tr.SendType = sendType
  204. tr.WxAppId = wxAppId
  205. go func() {
  206. err = models.AddUserTemplateRecord(tr)
  207. if err != nil {
  208. utils.FileLog.Info(fmt.Sprintf("AddUserTemplateRecord Err:%s", err.Error()))
  209. }
  210. }()
  211. }
  212. //accessToken过期
  213. if templateResponse.Errcode == 40001 {
  214. //强刷token并重新推送
  215. accessToken, err, errMsg = refreshWxAccessToken(wxAppId)
  216. if err != nil {
  217. utils.FileLog.Info(fmt.Sprintf("refreshWxAccessToken Err:%s", err.Error()))
  218. return err
  219. }
  220. return toSendTemplateMsg(wxAppId, data, resource, sendType, openId, uniqueCode)
  221. }
  222. //模板消息发送超过当日10万次限制错误处理
  223. if templateResponse.Errcode == 45009 {
  224. key := "CACHE_SendTemplateMsg_ERR"
  225. isExist := utils.Rc.IsExist(key)
  226. if isExist == true {
  227. return
  228. } else {
  229. result, _ := json.Marshal(templateResponse)
  230. if err != nil {
  231. utils.FileLog.Info(fmt.Sprintf("templateResponse Marshal Err:%s", err.Error()))
  232. return err
  233. }
  234. //发送邮件提醒异常
  235. go alarm_msg.SendAlarmMsg("模板消息发送超过当日10万次限制,templateResponse = "+string(result), 3)
  236. //go utils.SendEmail("异常提醒:", "模板消息发送超过当日10万次限制,templateResponse = "+string(result), utils.EmailSendToUsers)
  237. //设置3分钟缓存,不允许重复添加
  238. utils.Rc.SetNX(key, 1, 6*time.Minute)
  239. }
  240. //清空发送次数
  241. sendUrl := fmt.Sprintf("https://api.weixin.qq.com/cgi-bin/clear_quota?access_token=%s", accessToken)
  242. clearData := make(map[string]interface{})
  243. clearData["appid"] = utils.WxAppId
  244. clearJson, _ := json.Marshal(clearData)
  245. utils.FileLog.Info("clear_quota data:" + string(clearJson))
  246. resp, err := client.Post(sendUrl, "application/json", bytes.NewBuffer(clearJson))
  247. if err != nil {
  248. return err
  249. }
  250. defer resp.Body.Close()
  251. clearBody, err := ioutil.ReadAll(resp.Body)
  252. utils.FileLog.Info("clear_quota result:" + string(clearBody))
  253. var clearQuotaResponse ClearQuotaResponse
  254. err = json.Unmarshal(clearBody, &clearQuotaResponse)
  255. if err != nil {
  256. utils.FileLog.Info(fmt.Sprintf("clearQuotaResponse Unmarshal Err:%s", err.Error()))
  257. return err
  258. }
  259. if clearQuotaResponse.Errcode == 0 {
  260. //发送邮件解决异常
  261. go alarm_msg.SendAlarmMsg("异常已解决,自动清理限制接口,调用成功", 3)
  262. //go utils.SendEmail("异常已解决:", "自动清理限制接口,调用成功", utils.EmailSendToUsers)
  263. //重新推送一次
  264. toSendTemplateMsg(wxAppId, data, resource, sendType, openId, uniqueCode)
  265. }
  266. }
  267. return
  268. }
  269. // getWxAccessToken 获取微信token
  270. func getWxAccessToken(wxAppId string) (accessToken string, err error, errMsg string) {
  271. redisKey := getRedisKeyByAppid(wxAppId)
  272. if redisKey == `` {
  273. errMsg = "未配置缓存key"
  274. err = errors.New(errMsg)
  275. return
  276. }
  277. accessToken, err = utils.Rc.RedisString(redisKey)
  278. //fmt.Println(err)
  279. //fmt.Println(accessToken)
  280. //if err != nil {
  281. // errMsg = "GetWxAccessToken Err:" + err.Error()
  282. // utils.FileLog.Info("获取Token失败,msg:" + errMsg)
  283. // return
  284. //}
  285. //取到数据后就直接返回了,没有后续了
  286. if accessToken != "" {
  287. return
  288. }
  289. //缓存中没有取到数据,那么需要去强制刷新新的accessToken
  290. return refreshWxAccessToken(wxAppId)
  291. }
  292. // refreshWxAccessToken 强制刷新微信token
  293. func refreshWxAccessToken(wxAppId string) (accessToken string, err error, errMsg string) {
  294. fmt.Println("强制刷新" + wxAppId + "微信token")
  295. defer func() {
  296. if errMsg != `` {
  297. utils.FileLog.Info(fmt.Sprintf("强制刷新%s微信token异常:%s", wxAppId, errMsg))
  298. }
  299. }()
  300. redisKey := getRedisKeyByAppid(wxAppId)
  301. if redisKey == `` {
  302. errMsg = "未配置缓存key"
  303. err = errors.New(errMsg)
  304. return
  305. }
  306. wxAppSecret, ok := utils.WxAppList[wxAppId]
  307. if !ok {
  308. err = errors.New("缺少密钥信息")
  309. utils.FileLog.Info(fmt.Sprintf("获取Token失败, errMsg:%s", err.Error()))
  310. return
  311. }
  312. //调用微信官方接口获取新的accessToken
  313. wxAccessToken, tmpErr := models.GetWxToken(wxAppId, wxAppSecret)
  314. if tmpErr != nil {
  315. err = tmpErr
  316. errMsg = "通过微信接口获取accessToken失败 Err:" + err.Error()
  317. return
  318. }
  319. //如果没有token数据
  320. if wxAccessToken.AccessToken == "" {
  321. errMsg = "微信返回的accessToken异常: Err:" + wxAccessToken.Errmsg
  322. err = errors.New(errMsg)
  323. return
  324. }
  325. accessToken = wxAccessToken.AccessToken
  326. //如果是弘则研究的appid,那么需要更新mysql的accessToken
  327. if wxAppId == utils.WxAppId {
  328. expiresIn := time.Now().Add(time.Duration(wxAccessToken.ExpiresIn) * time.Second).Unix()
  329. err = models.ModifyAccessToken(wxAccessToken.AccessToken, expiresIn)
  330. if err != nil {
  331. errMsg = "更新mysql中的accessToken失败 Err:" + err.Error()
  332. return
  333. }
  334. }
  335. //更新redis的accessToken(过期时间提前十分钟)
  336. redisTimeExpire := time.Duration(wxAccessToken.ExpiresIn-600) * time.Second
  337. err = utils.Rc.Put(redisKey, accessToken, redisTimeExpire)
  338. if err != nil {
  339. errMsg = "更新redis中的accessToken失败 Err:" + err.Error()
  340. return
  341. }
  342. return
  343. }
  344. // WxGetRedisAccessToken 从redis中获取token
  345. func WxGetRedisAccessToken(wxAppId, wxAppSecret string) (accessToken string, err error) {
  346. //从redis中获取token校验验证码
  347. accessToken, err = utils.Rc.RedisString(utils.HZ_ADMIN_WX_ACCESS_TOEKN + wxAppId)
  348. if err != nil {
  349. err = nil
  350. token, tErr := models.GetWxToken(wxAppId, wxAppSecret)
  351. if tErr != nil {
  352. return "", tErr
  353. }
  354. if token.Errmsg != "" {
  355. err = errors.New("获取access_token 失败 errcode:" + token.Errmsg + " ;errmsg:" + token.Errmsg)
  356. return "", err
  357. }
  358. //更新redis的accessToken(过期时间提前十分钟)
  359. redisTimeExpire := time.Duration(token.ExpiresIn-600) * time.Second
  360. err = utils.Rc.Put(utils.HZ_ADMIN_WX_ACCESS_TOEKN+wxAppId, token.AccessToken, redisTimeExpire)
  361. if err != nil {
  362. err = errors.New("保存access_token失败 " + err.Error())
  363. return accessToken, err
  364. }
  365. accessToken = token.AccessToken
  366. return
  367. }
  368. return
  369. }
  370. // 根据微信appid获取对应的缓存key
  371. func getRedisKeyByAppid(wxAppId string) (redisKey string) {
  372. switch wxAppId {
  373. case utils.WxAppId:
  374. redisKey = utils.CACHE_WX_ACCESS_TOKEN_HZ
  375. case utils.AdminWxAppId:
  376. redisKey = utils.HZ_ADMIN_WX_ACCESS_TOEKN + wxAppId
  377. }
  378. return redisKey
  379. }