report_push.go 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. package services
  2. import (
  3. "context"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "github.com/wenzhenxi/gorsa"
  8. "hongze/hongze_task/models"
  9. "hongze/hongze_task/services/alarm_msg"
  10. "hongze/hongze_task/utils"
  11. "io/ioutil"
  12. "net/http"
  13. "net/url"
  14. "strconv"
  15. "time"
  16. )
  17. //func init() {
  18. // report, _ := models.GetReportById(836)
  19. // SendReportToThs(report)
  20. //}
  21. // SendWaitReport 定时向同花顺推送报告(定时任务)
  22. func SendWaitReport(cont context.Context) (err error) {
  23. defer func() {
  24. if err != nil {
  25. //fmt.Println(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "发送消息至同花顺失败 ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  26. //go utils.SendEmail(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "延时任务发送报告至同花顺失败 ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  27. go alarm_msg.SendAlarmMsg("延时任务发送报告至同花顺失败 ErrMsg:"+err.Error(), 3)
  28. }
  29. }()
  30. list, err := models.GetWaitReportSendThsDetailList()
  31. if err != nil {
  32. return
  33. }
  34. for _, detail := range list {
  35. //先标记为
  36. _ = models.ModifyReportSendThsDetailStatus(detail.SendId, 0, "")
  37. go SendToThs(detail.SendId, detail.ReportId, detail.ReportType)
  38. }
  39. return
  40. }
  41. // SyncWxGroupEveryDay 每日定时向同花顺同步客户时间
  42. func SyncWxGroupEveryDay(cont context.Context) (err error) {
  43. defer func() {
  44. if err != nil {
  45. fmt.Println(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "同步微信群组信息至同花顺失败 ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  46. //go utils.SendEmail(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "同步微信群组信息至同花顺失败 ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  47. }
  48. }()
  49. //查询校验当天数据是否已经生成
  50. _, err = models.GetCompanyEndDate(time.Now().Format(utils.FormatDate))
  51. if err == nil || err.Error() != utils.ErrNoRow() {
  52. err = errors.New(fmt.Sprint(time.Now().Format(utils.FormatDate), "当天数据已经同步"))
  53. return
  54. }
  55. //永续、正式、试用、冻结
  56. list, err := models.GetAllCompanyProduct()
  57. if err != nil {
  58. return
  59. }
  60. timeLoc, _ := time.LoadLocation("Asia/Shanghai")
  61. //需要入库的数据
  62. companyEndDateMap := make(map[string]*models.CompanyEndDate)
  63. //实际需要推送的数据
  64. companyEndDatePushMap := make(map[string]*models.CompanyEndDate)
  65. //微信群组开放编号code
  66. companyProductOpenCode := make(map[string]string)
  67. for _, companyProduct := range list {
  68. key := fmt.Sprint(companyProduct.CompanyId, "_", companyProduct.ProductId)
  69. endDate, _ := time.ParseInLocation(utils.FormatDate, companyProduct.EndDate, timeLoc)
  70. if companyProduct.Status == "冻结" {
  71. endDate = time.Now().AddDate(0, 0, -1)
  72. }
  73. tmpCompanyEndDate := &models.CompanyEndDate{
  74. CompanyId: companyProduct.CompanyId,
  75. ProductId: companyProduct.ProductId,
  76. Status: companyProduct.Status,
  77. EndDate: endDate,
  78. CreateTime: time.Now(),
  79. }
  80. companyEndDateMap[key] = tmpCompanyEndDate
  81. companyEndDatePushMap[key] = tmpCompanyEndDate
  82. companyProductOpenCode[key] = companyProduct.OpenCode
  83. }
  84. startDate := time.Now().AddDate(0, 0, -1).Format(utils.FormatDate)
  85. endDate := time.Now().Format(utils.FormatDate)
  86. endDateList, err := models.GetAllCompanyProductEndDate(startDate, endDate)
  87. for _, endDate := range endDateList {
  88. key := fmt.Sprint(endDate.CompanyId, "_", endDate.ProductId)
  89. if companyEndDate, ok := companyEndDatePushMap[key]; ok {
  90. if companyEndDate.Status == "冻结" {
  91. //获取当天的零点时间
  92. endDateStr := time.Now().AddDate(0, 0, -1).Format(utils.FormatDate)
  93. companyEndDate.EndDate, _ = time.ParseInLocation(utils.FormatDate, endDateStr, timeLoc)
  94. }
  95. if companyEndDate.EndDate.Equal(endDate.EndDate) {
  96. delete(companyEndDatePushMap, key)
  97. }
  98. }
  99. }
  100. for _, companyEndDate := range companyEndDatePushMap {
  101. fmt.Println("待推送数据:", companyEndDate)
  102. key := fmt.Sprint(companyEndDate.CompanyId, "_", companyEndDate.ProductId)
  103. //同步至同花顺
  104. if openCode, ok := companyProductOpenCode[key]; ok {
  105. SyncWxGroup(openCode, companyEndDate.EndDate.Format(utils.FormatDate))
  106. }
  107. }
  108. for _, companyEndDate := range companyEndDateMap {
  109. models.AddCompanyEndDate(companyEndDate)
  110. }
  111. return
  112. }
  113. var permissionMap map[string]string = map[string]string{
  114. "化里化外日评": "原油,PTA,MEG,织造终端,甲醇,聚烯烃,沥青,橡胶,苯乙烯,玻璃纯碱",
  115. "股债日评": "宏观,利率债,原油,PTA,MEG,织造终端,甲醇,聚烯烃,沥青,橡胶,苯乙烯,玻璃纯碱,钢材,铁矿,双焦(焦煤、焦炭),有色(铜、铝),有色(锌、铅),镍+不锈钢",
  116. "贵金属复盘": "宏观,利率债,原油,PTA,MEG,织造终端,甲醇,聚烯烃,沥青,橡胶,苯乙烯,玻璃纯碱,钢材,铁矿,双焦(焦煤、焦炭),有色(铜、铝),有色(锌、铅),镍+不锈钢",
  117. "每日经济数据备忘录": "宏观,利率债,原油,PTA,MEG,织造终端,甲醇,聚烯烃,沥青,橡胶,苯乙烯,玻璃纯碱,钢材,铁矿,双焦(焦煤、焦炭),有色(铜、铝),有色(锌、铅),镍+不锈钢",
  118. "宏观商品复盘": "宏观,利率债,原油,PTA,MEG,织造终端,甲醇,聚烯烃,沥青,橡胶,苯乙烯,玻璃纯碱,钢材,铁矿,双焦(焦煤、焦炭),有色(铜、铝),有色(锌、铅),镍+不锈钢",
  119. "知白守黑日评": "钢材,铁矿,双焦(焦煤、焦炭)",
  120. "有声有色日度闲篇": "有色(铜、铝),有色(锌、铅),镍+不锈钢",
  121. "EIA原油库存点评": "原油",
  122. "苯乙烯数据点评": "苯乙烯",
  123. "API原油库存点评": "原油",
  124. "铁矿航运数据点评": "铁矿",
  125. "中观需求点评": "宏观,利率债,原油,PTA,MEG,织造终端,甲醇,聚烯烃,沥青,橡胶,苯乙烯,玻璃纯碱,钢材,铁矿,双焦(焦煤、焦炭),有色(铜、铝),有色(锌、铅),镍+不锈钢",
  126. "聚酯数据点评": "PTA,MEG",
  127. "钢材周度数据点评": "钢材",
  128. "寻根知本": "宏观,利率债,原油,PTA,MEG,织造终端,甲醇,聚烯烃,沥青,橡胶,苯乙烯,玻璃纯碱,钢材,铁矿,双焦(焦煤、焦炭),有色(铜、铝),有色(锌、铅),镍+不锈钢",
  129. "国际宏观": "宏观,利率债,原油,PTA,MEG,织造终端,甲醇,聚烯烃,沥青,橡胶,苯乙烯,玻璃纯碱,钢材,铁矿,双焦(焦煤、焦炭),有色(铜、铝),有色(锌、铅),镍+不锈钢",
  130. "能化百家谈": "原油,PTA,MEG,织造终端,甲醇,聚烯烃,沥青,橡胶,苯乙烯,玻璃纯碱",
  131. "有色百家谈": "有色(铜、铝),有色(锌、铅),镍+不锈钢",
  132. "黑色百家谈": "钢材,铁矿,双焦(焦煤、焦炭)",
  133. }
  134. //TshResult 同花顺返回信息
  135. type TshResult struct {
  136. ErrorCode int `json:"error" description:"错误状态码"`
  137. Message string `json:"message" description:"提示信息"`
  138. }
  139. //
  140. func SendToThs(sendDetailId, reportId int, reportType string) (err error) {
  141. switch reportType {
  142. case "日度点评":
  143. tmpErr := SendReportToThs(reportId)
  144. if tmpErr != nil {
  145. err = tmpErr
  146. }
  147. case "研报小程序":
  148. tmpErr := SendReportMiniToThs(reportId)
  149. if tmpErr != nil {
  150. err = tmpErr
  151. }
  152. default:
  153. err = errors.New("异常类型")
  154. }
  155. if err != nil {
  156. _ = models.ModifyReportSendThsDetailStatus(sendDetailId, -1, err.Error())
  157. err = nil
  158. return
  159. }
  160. _ = models.ModifyReportSendThsDetailStatus(sendDetailId, 1, "")
  161. return
  162. }
  163. // SendReportToThs 发送报告到同花顺
  164. func SendReportToThs(reportId int) (err error) {
  165. defer func() {
  166. if err != nil {
  167. //fmt.Println(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "发送消息至同花顺失败 ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  168. //go utils.SendEmail(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "发送报告至同花顺失败 ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  169. go alarm_msg.SendAlarmMsg("发送报告至同花顺失败 ErrMsg:"+err.Error(), 3)
  170. }
  171. }()
  172. report, tmpErr := models.GetReportById(reportId)
  173. if tmpErr != nil {
  174. err = tmpErr
  175. return
  176. }
  177. //获取分类信息(标签)
  178. permissionName := report.ClassifyNameSecond
  179. classifyItem, err := models.GetClassifyById(report.ClassifyIdSecond)
  180. if err != nil {
  181. err = errors.New(fmt.Sprint("获取分类失败:", permissionName))
  182. }
  183. //获取权限标签名称
  184. var permissionStr string
  185. if classifyItem != nil {
  186. permissionStr = classifyItem.ClassifyLabel
  187. if permissionStr == "" {
  188. var isOk bool
  189. permissionStr, isOk = permissionMap[permissionName]
  190. if !isOk {
  191. err = errors.New(fmt.Sprint("没有该权限的标签,权限名:", permissionName))
  192. return
  193. }
  194. }
  195. } else {
  196. var isOk bool
  197. permissionStr, isOk = permissionMap[permissionName]
  198. if !isOk {
  199. err = errors.New(fmt.Sprint("没有该权限的标签,权限名:", permissionName))
  200. return
  201. }
  202. }
  203. if permissionStr == "" {
  204. err = errors.New(fmt.Sprint("没有该权限的标签,权限名:", permissionName))
  205. return
  206. }
  207. //fmt.Println("sendDetailId:", sendDetailId)
  208. stageStr := fmt.Sprintf("%v", report.Stage)
  209. createDate, err := time.Parse(utils.FormatDateTime, report.CreateTime)
  210. createDateFrom := createDate.Format("0102")
  211. title := `【第` + stageStr + `期|FICC】` + report.Title + `(` + createDateFrom + ")"
  212. // 跳转地址
  213. jumpBaseUrl := `http://rddpweb.brilliantstart.cn/reportdtl?id=`
  214. //生产环境地址
  215. if utils.RunMode == "release" {
  216. jumpBaseUrl = `https://ficc.hzinsights.com/reportdtl?id=`
  217. }
  218. jumpUrl := fmt.Sprint(jumpBaseUrl, report.Id)
  219. // logo 地址
  220. logoUrl := `https://hongze.oss-cn-shanghai.aliyuncs.com/hzyj.png`
  221. // 简介
  222. abstract := report.Abstract
  223. // 实际推送到同花顺
  224. err = SendThs(title, permissionStr, abstract, jumpUrl, logoUrl, "1")
  225. return
  226. }
  227. // SendThs 发送消息到同花顺
  228. func SendThs(title, labelStr, abstract, jumpBaseUrl, logoUrl, dataType string) (err error) {
  229. defer func() {
  230. if err != nil {
  231. //fmt.Println(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "发送消息至同花顺失败 ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  232. //go utils.SendEmail(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "发送消息至同花顺失败 ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  233. go alarm_msg.SendAlarmMsg("发送消息至同花顺失败 ErrMsg:"+err.Error(), 3)
  234. }
  235. }()
  236. pubKey := utils.THS_PubKey
  237. sendUrl := utils.THS_SendUrl
  238. //fmt.Println("sendUrl:", sendUrl)
  239. //标题字符长度截取,最多50位字符
  240. title = utils.SubStr(title, 50)
  241. utils.FileLog.Info(fmt.Sprintf("title:%s", title))
  242. title, err = gorsa.PublicEncrypt(title, pubKey)
  243. if err != nil {
  244. return
  245. }
  246. //简介字符长度截取,最多50位字符
  247. abstract = utils.SubStr(abstract, 50)
  248. utils.FileLog.Info(fmt.Sprintf("abstract:%s", abstract))
  249. abstract, err = gorsa.PublicEncrypt(abstract, pubKey)
  250. if err != nil {
  251. return
  252. }
  253. utils.FileLog.Info(fmt.Sprintf("labelStr:%s", labelStr))
  254. label, err := gorsa.PublicEncrypt(labelStr, pubKey)
  255. if err != nil {
  256. return
  257. }
  258. jumpUrl, err := gorsa.PublicEncrypt(jumpBaseUrl, pubKey)
  259. if err != nil {
  260. return
  261. }
  262. picUrl, err := gorsa.PublicEncrypt(logoUrl, pubKey)
  263. if err != nil {
  264. return
  265. }
  266. dataTypeEncript, err := gorsa.PublicEncrypt(dataType, pubKey)
  267. if err != nil {
  268. return
  269. }
  270. //开始发送
  271. client := http.Client{}
  272. form := url.Values{}
  273. form.Add("title", title)
  274. form.Add("description", abstract)
  275. form.Add("label", label)
  276. form.Add("url", jumpUrl)
  277. form.Add("icon", picUrl)
  278. form.Add("dataType", dataTypeEncript)
  279. utils.FileLog.Info(fmt.Sprintf("SendThs parms:%s", form.Encode()))
  280. resp, err := client.PostForm(sendUrl, form)
  281. if err != nil {
  282. return
  283. }
  284. defer resp.Body.Close()
  285. body, _ := ioutil.ReadAll(resp.Body)
  286. //fmt.Println(string(body))
  287. utils.FileLog.Info(fmt.Sprintf("ThsResult parms:%s", string(body)))
  288. //同花顺接口返回数据
  289. var tshResult TshResult
  290. err = json.Unmarshal(body, &tshResult)
  291. if err != nil {
  292. err = errors.New(fmt.Sprint("同花顺接口返回数据转换成结构体异常,Err:", err))
  293. return
  294. }
  295. if tshResult.ErrorCode != 1 {
  296. err = errors.New(fmt.Sprint("发送数据到同花顺接口异常,result:", string(body)))
  297. return
  298. }
  299. return
  300. }
  301. // SyncWxGroup 同步同花顺 微信群信息
  302. func SyncWxGroup(openCompanyCode, deadline string) (err error) {
  303. defer func() {
  304. if err != nil {
  305. //fmt.Println(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "同步同花顺微信群信息失败 ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  306. //go utils.SendEmail(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "同步同花顺微信群信息失败 ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  307. go alarm_msg.SendAlarmMsg("同步同花顺微信群信息失败 ErrMsg:"+err.Error(), 3)
  308. }
  309. }()
  310. pubKey := utils.THS_PubKey
  311. sendUrl := utils.THS_SyncWxGroupUrl
  312. openCompanyCode, err = gorsa.PublicEncrypt(openCompanyCode, pubKey)
  313. if err != nil {
  314. return
  315. }
  316. deadline, err = gorsa.PublicEncrypt(deadline, pubKey)
  317. if err != nil {
  318. return
  319. }
  320. status := `1`
  321. status, err = gorsa.PublicEncrypt(status, pubKey)
  322. if err != nil {
  323. return
  324. }
  325. //开始发送
  326. client := http.Client{}
  327. form := url.Values{}
  328. form.Add("thirdWechatGroupId", openCompanyCode)
  329. form.Add("deadline", deadline)
  330. form.Add("status", status)
  331. utils.FileLog.Info(fmt.Sprintf("SendThs SyncWxGroup parms:%s", form.Encode()))
  332. resp, err := client.PostForm(sendUrl, form)
  333. if err != nil {
  334. return
  335. }
  336. defer resp.Body.Close()
  337. body, _ := ioutil.ReadAll(resp.Body)
  338. fmt.Println(string(body))
  339. utils.FileLog.Info(fmt.Sprintf("SyncWxGroup ThsResult parms:%s", string(body)))
  340. //同花顺接口返回数据
  341. var tshResult TshResult
  342. err = json.Unmarshal(body, &tshResult)
  343. if err != nil {
  344. err = errors.New(fmt.Sprint("同花顺接口返回数据转换成结构体异常,Err:", err))
  345. return
  346. }
  347. if tshResult.ErrorCode != 1 {
  348. err = errors.New(fmt.Sprint("发送数据到同花顺接口异常,result:", string(body)))
  349. return
  350. }
  351. return
  352. }
  353. // SendReportMiniToThs 发送报告小程序到同花顺
  354. func SendReportMiniToThs(reportId int) (err error) {
  355. defer func() {
  356. if err != nil {
  357. //fmt.Println(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "发送消息至同花顺失败 ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  358. //go utils.SendEmail(utils.APPNAME+"【"+utils.RunMode+"】"+"失败提醒", "SendReportMiniToThs延时任务发送报告至同花顺失败, ReportId:"+strconv.Itoa(reportId)+", ErrMsg:"+err.Error(), utils.EmailSendToUsers)
  359. go alarm_msg.SendAlarmMsg("SendReportMiniToThs延时任务发送报告至同花顺失败, ReportId:"+strconv.Itoa(reportId)+", ErrMsg:"+err.Error(), 3)
  360. }
  361. }()
  362. report, tmpErr := models.GetReportById(reportId)
  363. if tmpErr != nil {
  364. err = tmpErr
  365. return
  366. }
  367. //小程序跳转地址
  368. jumpBaseUrl := utils.WxYbAppId + `/pages-report/reportDetail?reportId=`
  369. jumpUrl := fmt.Sprint(jumpBaseUrl, report.Id)
  370. logoUrl := `https://hongze.oss-cn-shanghai.aliyuncs.com/hzyj.png`
  371. var permissionStr string
  372. if report.HasChapter == 0 {
  373. // 获取分类信息(标签)
  374. permissionName := report.ClassifyNameSecond
  375. classifyItem, tmpErr := models.GetClassifyById(report.ClassifyIdSecond)
  376. if tmpErr != nil {
  377. err = errors.New(fmt.Sprint("获取分类失败:", permissionName))
  378. return
  379. }
  380. // 获取权限标签名称
  381. if classifyItem != nil {
  382. permissionStr = classifyItem.ClassifyLabel
  383. if permissionStr == "" {
  384. var isOk bool
  385. permissionStr, isOk = permissionMap[permissionName]
  386. if !isOk {
  387. err = errors.New(fmt.Sprint("没有该权限的标签,权限名:", permissionName))
  388. return
  389. }
  390. }
  391. } else {
  392. var isOk bool
  393. permissionStr, isOk = permissionMap[permissionName]
  394. if !isOk {
  395. err = errors.New(fmt.Sprint("没有该权限的标签,权限名:", permissionName))
  396. return
  397. }
  398. }
  399. if permissionStr == "" {
  400. err = errors.New(fmt.Sprint("没有该权限的标签,权限名:", permissionName))
  401. return
  402. }
  403. } else {
  404. // 同php的
  405. permissionStr = "宏观,利率债,原油,PTA,MEG,织造终端,甲醇,聚烯烃,沥青,橡胶,苯乙烯,玻璃纯碱,钢材,铁矿,双焦(焦煤、焦炭),有色(铜、铝),有色(锌、铅),镍+不锈钢"
  406. }
  407. // 实际推送到同花顺
  408. err = SendThs(report.Title, permissionStr, report.Abstract, jumpUrl, logoUrl, "2")
  409. return
  410. }