report_push.go 12 KB

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