report_service.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611
  1. package report
  2. import (
  3. "encoding/json"
  4. "eta/eta_mini_ht_api/common/component/config"
  5. "eta/eta_mini_ht_api/common/component/es"
  6. logger "eta/eta_mini_ht_api/common/component/log"
  7. "eta/eta_mini_ht_api/common/contants"
  8. "eta/eta_mini_ht_api/common/utils/page"
  9. stringUtils "eta/eta_mini_ht_api/common/utils/string"
  10. analystService "eta/eta_mini_ht_api/domian/financial_analyst"
  11. userService "eta/eta_mini_ht_api/domian/user"
  12. "eta/eta_mini_ht_api/models"
  13. "eta/eta_mini_ht_api/models/eta"
  14. etaDao "eta/eta_mini_ht_api/models/eta"
  15. "eta/eta_mini_ht_api/models/ht"
  16. mediaDao "eta/eta_mini_ht_api/models/media"
  17. reportDao "eta/eta_mini_ht_api/models/report"
  18. "github.com/google/uuid"
  19. "math/rand"
  20. "strconv"
  21. "strings"
  22. "time"
  23. )
  24. const (
  25. SourceETA = "ETA"
  26. SourceHT = "HT"
  27. DESC models.Order = "desc"
  28. ASC models.Order = "asc"
  29. ESColumn = "title"
  30. ESRangeColumn = "reportId"
  31. )
  32. var (
  33. sortField = []string{"_score:desc"}
  34. htConfig = config.GetConfig(contants.HT).(*config.HTBizConfig)
  35. )
  36. func elastic() *es.ESClient {
  37. return es.GetInstance()
  38. }
  39. // ESReport Report ES研报mapping
  40. type ESReport struct {
  41. ReportID int `json:"reportId"`
  42. OrgId int `json:"orgId"`
  43. Title string `json:"title"`
  44. Author string `json:"author"`
  45. Source reportDao.ReportSource `json:"source"`
  46. Abstract string `json:"abstract"`
  47. CoverSrc int `json:"coverSrc"`
  48. Status reportDao.ReportStatus `json:"status"`
  49. PublishedTime string `json:"publishedTime"`
  50. }
  51. type ReportDTO struct {
  52. ReportID int `json:"reportId"`
  53. OrgId int `json:"orgId"`
  54. Title string `json:"title"`
  55. Author string `json:"author"`
  56. Source string `json:"source"`
  57. Abstract string `json:"abstract"`
  58. PublishedTime string `json:"publishedTime"`
  59. SecondPermission map[int]string `json:"-"`
  60. Permissions map[int]string `json:"-"`
  61. PermissionNames interface{} `json:"permissionNames,omitempty"`
  62. Highlight []string `json:"highlight,omitempty"`
  63. Detail json.RawMessage `json:"detail,omitempty"`
  64. PdfUrl string `json:"pdfUrl,omitempty"`
  65. CoverSrc int `json:"-"`
  66. CoverUrl string `json:"coverUrl"`
  67. }
  68. type Detail struct {
  69. }
  70. type PermissionDTO struct {
  71. ID int
  72. Name string
  73. ParentID int
  74. }
  75. func GetGetReportById(reportId int) (ReportDTO ReportDTO, err error) {
  76. report, err := reportDao.GetReportById(reportId)
  77. if err != nil {
  78. return
  79. }
  80. orgId := report.OrgID
  81. names, _ := reportDao.GetAuthorByOrgId(orgId, string(report.Source))
  82. if names != nil && len(names) > 1 {
  83. names = stringUtils.UniqueItems(names)
  84. report.Author = strings.Join(names, ",")
  85. }
  86. report.PublishedTime = report.PublishedTime[:10]
  87. ReportDTO = convertReportDTO(report)
  88. return
  89. }
  90. func GetTotalPageCount() (total int64) {
  91. return reportDao.GetTotalPageCount()
  92. }
  93. func GetTotalPageCountByAnalyst(analyst string) (total int64, latestId int64) {
  94. return reportDao.GetTotalPageCountByAnalyst(analyst)
  95. }
  96. func SearchMaxReportId(key string) (total int64, reportId int64) {
  97. sort := []string{"reportId:desc"}
  98. request := matchAll(sort, key)
  99. //同步es
  100. re, err := elastic().Count(request)
  101. if err != nil {
  102. logger.Error("es搜索异常:%v", err)
  103. }
  104. count := re.Count
  105. total = int64(count)
  106. if total > 0 {
  107. request = match(key, 0, count, sort)
  108. re, err = elastic().Search(request)
  109. if err != nil {
  110. logger.Error("es搜索异常:%v", err)
  111. }
  112. hits := elastic().GetSource(re.Hits)
  113. data := hits[0].Source
  114. report := ReportDTO{}
  115. err = json.Unmarshal(data, &report)
  116. if err != nil {
  117. logger.Error("获取当前最大研报id失败:%v", err)
  118. return
  119. }
  120. reportId = int64(report.ReportID)
  121. }
  122. return
  123. }
  124. func SearchReportList(key string, from int, size int, max int64) (reports []ReportDTO, err error) {
  125. //同步es
  126. sorts := append(sortField, "publishedTime:desc")
  127. request := matchRange(key, from, size, max, sorts)
  128. re, err := elastic().Search(request)
  129. if err != nil {
  130. logger.Error("es搜索异常:%v", err)
  131. }
  132. hits := elastic().GetSource(re.Hits)
  133. if len(hits) == 0 {
  134. reports = []ReportDTO{}
  135. return
  136. }
  137. for _, hit := range hits {
  138. var content map[string][]string
  139. err = json.Unmarshal(hit.Highlight, &content)
  140. report := ReportDTO{}
  141. err = json.Unmarshal(hit.Source, &report)
  142. if err != nil {
  143. logger.Error("解析研报数据失败:%v", err)
  144. continue
  145. }
  146. report.Highlight = content[ESColumn]
  147. report.Title = report.Highlight[0]
  148. report.PublishedTime = report.PublishedTime[:10]
  149. reports = append(reports, report)
  150. }
  151. return
  152. }
  153. func GetReportPageByAnalyst(pageInfo page.PageInfo, analyst string) (list []ReportDTO, err error) {
  154. offset := page.StartIndex(pageInfo.Current, pageInfo.PageSize)
  155. reports, err := reportDao.GetReportPageByAnalyst(pageInfo.LatestId, pageInfo.PageSize, offset, analyst)
  156. if err != nil {
  157. logger.Error("分页查询报告列表失败:%v", err)
  158. return
  159. }
  160. list = make([]ReportDTO, 0)
  161. if reports != nil {
  162. for _, report := range reports {
  163. dto := convertReportDTO(report)
  164. list = append(list, dto)
  165. }
  166. }
  167. return
  168. }
  169. func GetReportPageByOrgIds(pageInfo page.PageInfo, orgIds map[string][]int) (list []ReportDTO, err error) {
  170. offset := page.StartIndex(pageInfo.Current, pageInfo.PageSize)
  171. reports, err := reportDao.GetReportPageByOrgIds(pageInfo.LatestId, pageInfo.PageSize, offset, orgIds)
  172. if err != nil {
  173. logger.Error("分页查询报告列表失败:%v", err)
  174. return
  175. }
  176. list = make([]ReportDTO, 0)
  177. if reports != nil {
  178. for _, report := range reports {
  179. dto := convertReportDTO(report)
  180. list = append(list, dto)
  181. }
  182. }
  183. return
  184. }
  185. func GetNewReportByPublishTime(time time.Time) (reports []ReportDTO) {
  186. list := reportDao.GetNewReportByPublishTime(time)
  187. if list != nil {
  188. for _, report := range list {
  189. dto := convertReportDTO(report)
  190. reports = append(reports, dto)
  191. }
  192. }
  193. return
  194. }
  195. func GetReportPage(pageInfo page.PageInfo) (list []ReportDTO, err error) {
  196. offset := page.StartIndex(pageInfo.Current, pageInfo.PageSize)
  197. reports, err := reportDao.GetReportPage(pageInfo.LatestId, pageInfo.PageSize, offset)
  198. if err != nil {
  199. logger.Error("分页查询报告列表失败:%v", err)
  200. return
  201. }
  202. list = make([]ReportDTO, 0)
  203. if reports != nil {
  204. for _, report := range reports {
  205. dto := convertReportDTO(report)
  206. list = append(list, dto)
  207. }
  208. }
  209. return
  210. }
  211. func getETAReportFirstPermissions(id int) (permissionDTOs []PermissionDTO) {
  212. classifyId, err := etaDao.GetReportClassifyById(id)
  213. if err != nil || classifyId == 0 {
  214. logger.Error("获取研报分类信息失败:%v", err)
  215. return
  216. }
  217. permissions, err := etaDao.GetFirstPermissionsByClassifyID(classifyId)
  218. if err != nil {
  219. logger.Error("获取研报一级品种信息失败:%v", err)
  220. return
  221. }
  222. for _, permission := range permissions {
  223. permissionDTOs = append(permissionDTOs, convertPermissionDTO(permission))
  224. }
  225. return
  226. }
  227. func getETAReportSecondPermissions(id int) (permissionDTOs []PermissionDTO) {
  228. classifyId, err := etaDao.GetReportClassifyById(id)
  229. if err != nil || classifyId == 0 {
  230. logger.Error("获取研报分类信息失败:%v", err)
  231. return
  232. }
  233. permissions, err := etaDao.GetSecondPermissionsByClassifyID(classifyId)
  234. if err != nil {
  235. logger.Error("获取研报二级品种信息失败:%v", err)
  236. return
  237. }
  238. for _, permission := range permissions {
  239. permissionDTOs = append(permissionDTOs, convertPermissionDTO(permission))
  240. }
  241. return
  242. }
  243. func (es ESReport) GetId() string {
  244. return strconv.Itoa(es.ReportID)
  245. }
  246. func GetETALatestReportId() (id int, err error) {
  247. return reportDao.GetLatestReportIdBySource(reportDao.SourceETA)
  248. }
  249. func GetHTLatestReportId() (id int, err error) {
  250. return reportDao.GetLatestReportIdBySource(reportDao.SourceHT)
  251. }
  252. func InitETAReportList(list []eta.ETAReport) (err error) {
  253. logger.Info("同步研报数量%d", len(list))
  254. var reports []reportDao.Report
  255. for _, etaRp := range list {
  256. authorNames := strings.Split(etaRp.Author, ",")
  257. authorNamesWithOutEmpty := stringUtils.RemoveEmptyStrings(authorNames)
  258. for _, authorName := range authorNamesWithOutEmpty {
  259. var coverSrc int
  260. var permissions []etaDao.ChartPermission
  261. permissions, err = etaDao.GetSecondPermissionsByClassifyID(etaRp.ClassifyID)
  262. if err != nil || len(permissions) == 0 {
  263. logger.Error("获取研报二级品种信息失败:%v", err)
  264. coverSrc = 0
  265. } else {
  266. permissionsId := permissions[0].ChartPermissionID
  267. var ids []int
  268. ids, err = mediaDao.GetIdsByPermissionId(permissionsId)
  269. if err != nil {
  270. logger.Error("获取图片资源失败:%v", err)
  271. }
  272. if ids == nil || len(ids) == 0 {
  273. coverSrc = 0
  274. } else {
  275. src := rand.NewSource(time.Now().UnixNano())
  276. r := rand.New(src)
  277. // 从切片中随机选择一个元素
  278. randomIndex := r.Intn(len(ids))
  279. coverSrc = ids[randomIndex]
  280. }
  281. }
  282. destRp := convertEtaReport(etaRp)
  283. destRp.Author = authorName
  284. destRp.CoverSrc = coverSrc
  285. reports = append(reports, destRp)
  286. }
  287. }
  288. err = reportDao.BatchInsertReport(&reports)
  289. if err != nil {
  290. logger.Error("同步ETA研报失败:%v", err)
  291. return
  292. }
  293. return syncES(reports)
  294. }
  295. func SyncETAReportList(list []eta.ETAReport) (err error) {
  296. logger.Info("同步研报数量%d", len(list))
  297. var reports []reportDao.Report
  298. for _, etaRp := range list {
  299. authorNames := strings.Split(etaRp.Author, ",")
  300. authorNamesWithOutEmpty := stringUtils.RemoveEmptyStrings(authorNames)
  301. for _, authorName := range authorNamesWithOutEmpty {
  302. destRp := convertEtaReport(etaRp)
  303. destRp.Author = authorName
  304. reports = append(reports, destRp)
  305. }
  306. }
  307. err = reportDao.BatchInsertReport(&reports)
  308. if err != nil {
  309. logger.Error("同步ETA研报失败:%v", err)
  310. return
  311. }
  312. return syncESAndSendMessage(reports)
  313. }
  314. func syncESAndSendMessage(reports []reportDao.Report) (err error) {
  315. var esReports []es.ESBase
  316. for _, etaRp := range reports {
  317. esRp := convertEsReport(etaRp)
  318. esReports = append(esReports, esRp)
  319. }
  320. //同步es
  321. err = elastic().BulkInsert(htConfig.GetReportIndex(), esReports)
  322. if err != nil {
  323. logger.Error("同步ETA研报到es失败:%v", err)
  324. return
  325. }
  326. //生产meta信息
  327. logger.Info("生成推送META信息")
  328. for _, report := range reports {
  329. userIds := userService.GetPostUser(report.Author, report.PublishedTime)
  330. var author analystService.FinancialAnalystDTO
  331. author, err = analystService.GetAnalystByName(report.Author)
  332. if err != nil {
  333. logger.Error("获取研报作者失败:%v", err)
  334. continue
  335. }
  336. if len(userIds) > 0 {
  337. usersStr := stringUtils.IntToStringSlice(userIds)
  338. Meta := userService.MetaData{
  339. AuthorName: report.Author,
  340. AuthorId: author.Id,
  341. SourceId: report.ID,
  342. PublishedTime: report.PublishedTime,
  343. }
  344. metaStr, _ := json.Marshal(Meta)
  345. toStr := strings.Join(usersStr, ",")
  346. UUID := uuid.New()
  347. uuidStr := UUID.String()
  348. metaContent := userService.MetaInfoDTO{
  349. From: "HT",
  350. Uid: "report:" + uuidStr,
  351. Meta: string(metaStr),
  352. MetaType: "USER_NOTICE",
  353. SourceType: "REPORT",
  354. To: toStr,
  355. }
  356. err = userService.CreateMetaInfo(metaContent)
  357. if err != nil {
  358. logger.Error("创建Meta信息失败:%v", err)
  359. return err
  360. }
  361. }
  362. }
  363. return
  364. }
  365. func syncES(reports []reportDao.Report) (err error) {
  366. var esReports []es.ESBase
  367. for _, etaRp := range reports {
  368. esRp := convertEsReport(etaRp)
  369. esReports = append(esReports, esRp)
  370. }
  371. //同步es
  372. err = elastic().BulkInsert(htConfig.GetReportIndex(), esReports)
  373. if err != nil {
  374. logger.Error("同步ETA研报到es失败:%v", err)
  375. return
  376. }
  377. return
  378. }
  379. func InitHTReportList(list []ht.HTReport) (noRecord bool, err error) {
  380. var reports []reportDao.Report
  381. //获取系统中ht品种名
  382. permissions, err := reportDao.GetGLAuthorNames()
  383. if err != nil {
  384. logger.Error("获取钢联研报作者失败:%v", err)
  385. return
  386. }
  387. for _, htRp := range list {
  388. for _, permission := range permissions {
  389. if htRp.PermissionName == permission.Permission {
  390. if permission.AuthorNames != "" {
  391. htRp.PublishUserName = permission.AuthorNames
  392. }
  393. authorNames := strings.Split(htRp.PublishUserName, ",")
  394. authorNamesWithOutEmpty := stringUtils.RemoveEmptyStrings(authorNames)
  395. for _, authorName := range authorNamesWithOutEmpty {
  396. destRp := convertHTReport(htRp)
  397. destRp.Author = authorName
  398. var coverSrc int
  399. permissionId, err := etaDao.GetPermissionIdByName(htRp.PermissionName)
  400. if err != nil {
  401. logger.Error("HT获取eta品种id失败:%v", err)
  402. coverSrc = 0
  403. }
  404. ids, err := mediaDao.GetIdsByPermissionId(permissionId)
  405. if err != nil {
  406. logger.Error("获取图片资源失败:%v", err)
  407. coverSrc = 0
  408. }
  409. if ids == nil || len(ids) == 0 {
  410. coverSrc = 0
  411. } else {
  412. src := rand.NewSource(time.Now().UnixNano())
  413. r := rand.New(src)
  414. // 从切片中随机选择一个元素
  415. randomIndex := r.Intn(len(ids))
  416. coverSrc = ids[randomIndex]
  417. }
  418. destRp.CoverSrc = coverSrc
  419. destRp.PlateName = htRp.PermissionName
  420. reports = append(reports, destRp)
  421. }
  422. }
  423. }
  424. }
  425. if len(reports) == 0 {
  426. return true, nil
  427. } else {
  428. logger.Info("同步研报数量%d", len(reports))
  429. err = reportDao.BatchInsertReport(&reports)
  430. if err != nil {
  431. logger.Error("同步HT研报失败:%v", err)
  432. return false, err
  433. }
  434. return false, syncES(reports)
  435. }
  436. }
  437. func SyncHTReportList(list []ht.HTReport) (err error) {
  438. //logger.Info("同步研报数量%d", len(list))
  439. //var reports []reportDao.Report
  440. //for _, htRp := range list {
  441. // var authorStr string
  442. // authorStr, err = reportDao.GetGLAuthorNames(htRp.Plate, htRp.Permission)
  443. // if err != nil {
  444. // logger.Error("获取钢联研报作者失败:%v", err)
  445. // }
  446. // if authorStr != "" {
  447. // htRp.Author = authorStr
  448. // }
  449. // authorNames := strings.Split(htRp.Author, ",")
  450. // authorNamesWithOutEmpty := stringUtils.RemoveEmptyStrings(authorNames)
  451. // for _, authorName := range authorNamesWithOutEmpty {
  452. // destRp := convertHTReport(htRp)
  453. // destRp.Author = authorName
  454. // reports = append(reports, destRp)
  455. // }
  456. //}
  457. //err = reportDao.BatchInsertReport(&reports)
  458. //if err != nil {
  459. // logger.Error("同步HT研报失败:%v", err)
  460. // return
  461. //}
  462. //return syncESAndSendMessage(reports)
  463. return
  464. }
  465. func GetListOrderByConditionWeekly(week bool, column string, limit int, order models.Order) (dtoList []ReportDTO, err error) {
  466. reports, err := reportDao.GetListOrderByCondition(week, column, limit, order)
  467. if err != nil {
  468. logger.Error("获取研报失败:%v", err)
  469. return
  470. }
  471. for _, report := range reports {
  472. dto := convertReportDTO(report)
  473. dtoList = append(dtoList, dto)
  474. }
  475. return
  476. }
  477. func GetListByCondition[T any](column string, ids []T) (dtoList []ReportDTO, err error) {
  478. var values []interface{}
  479. for _, id := range ids {
  480. values = append(values, id)
  481. }
  482. reports, err := reportDao.GetListByCondition(column, ids)
  483. if err != nil {
  484. logger.Error("获取研报失败:%v", err)
  485. return
  486. }
  487. for _, report := range reports {
  488. dto := convertReportDTO(report)
  489. dtoList = append(dtoList, dto)
  490. }
  491. return
  492. }
  493. func GetTotalPageCountByPermissionIds(permissionIds []int) (total int64, latestId int64, ids map[string][]int) {
  494. htIds, err := GetHTReportIdsByPermissionIds(permissionIds)
  495. if err != nil {
  496. logger.Error("品种筛选ht报告id失败:%v", err)
  497. htIds = []int{}
  498. }
  499. etaIds, err := GetETAReportIdsByPermissionIds(permissionIds)
  500. if err != nil {
  501. logger.Error("品种筛选eta报告id失败:%v", err)
  502. etaIds = []int{}
  503. }
  504. total = int64(len(etaIds) + len(htIds))
  505. ids = make(map[string][]int, 2)
  506. if len(etaIds) == 0 {
  507. ids["ETA"] = []int{}
  508. } else {
  509. ids["ETA"] = etaIds
  510. }
  511. if len(htIds) == 0 {
  512. ids["HT"] = []int{}
  513. } else {
  514. ids["HT"] = htIds
  515. }
  516. //ids = append(etaIds, htIds...)
  517. latestId = reportDao.GetMaxIdByPermissionIds(ids)
  518. return
  519. }
  520. func convertEtaReport(etaRp eta.ETAReport) reportDao.Report {
  521. return reportDao.Report{
  522. OrgID: etaRp.ID,
  523. Title: etaRp.Title,
  524. Abstract: etaRp.Abstract,
  525. Author: etaRp.Author,
  526. CoverSrc: 0,
  527. PublishedTime: etaRp.PublishTime,
  528. Source: reportDao.SourceETA,
  529. SendStatus: reportDao.UNSEND,
  530. Status: reportDao.StatusInit,
  531. }
  532. }
  533. func convertHTReport(etaRp ht.HTReport) reportDao.Report {
  534. return reportDao.Report{
  535. OrgID: etaRp.Id,
  536. Title: etaRp.ReportName,
  537. Author: etaRp.PublishUserName,
  538. PublishedTime: etaRp.PublishedTime,
  539. CoverSrc: 0,
  540. Source: reportDao.SourceHT,
  541. SendStatus: reportDao.UNSEND,
  542. Status: reportDao.StatusInit,
  543. }
  544. }
  545. func convertEsReport(report reportDao.Report) ESReport {
  546. return ESReport{
  547. ReportID: report.ID,
  548. Title: report.Title,
  549. OrgId: report.OrgID,
  550. Author: report.Author,
  551. Source: report.Source,
  552. Abstract: report.Abstract,
  553. Status: report.Status,
  554. CoverSrc: report.CoverSrc,
  555. PublishedTime: report.PublishedTime,
  556. }
  557. }
  558. func convertReportDTO(report reportDao.Report) (reportDTO ReportDTO) {
  559. reportDTO = ReportDTO{
  560. ReportID: report.ID,
  561. Title: report.Title,
  562. OrgId: report.OrgID,
  563. Author: report.Author,
  564. Source: string(report.Source),
  565. CoverSrc: report.CoverSrc,
  566. Abstract: report.Abstract,
  567. PublishedTime: report.PublishedTime,
  568. }
  569. publishDate, err := time.Parse(time.DateTime, reportDTO.PublishedTime)
  570. if err == nil {
  571. reportDTO.PublishedTime = publishDate.Format(time.DateOnly)
  572. }
  573. return
  574. }
  575. func matchAll(sorts []string, key string) (request *es.ESQueryRequest) {
  576. req := new(es.ESQueryRequest)
  577. return req.CreateESQueryRequest(htConfig.GetReportIndex(), ESColumn, key, 0, 1, sorts, es.MatchAll)
  578. }
  579. func match(key string, from int, to int, sorts []string) (request *es.ESQueryRequest) {
  580. req := new(es.ESQueryRequest)
  581. return req.CreateESQueryRequest(htConfig.GetReportIndex(), ESColumn, key, from, to, sorts, es.Match)
  582. }
  583. func matchRange(key string, from int, to int, max int64, sorts []string) (request *es.ESQueryRequest) {
  584. req := new(es.ESQueryRequest)
  585. return req.CreateESQueryRequest(htConfig.GetReportIndex(), ESColumn, key, from, to, sorts, es.Range).Range(0, max, ESRangeColumn)
  586. }