match_active.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537
  1. package service
  2. import (
  3. "context"
  4. "time"
  5. "go-common/app/interface/main/esports/model"
  6. arcmdl "go-common/app/service/main/archive/api"
  7. "go-common/library/ecode"
  8. "go-common/library/log"
  9. "go-common/library/sync/errgroup"
  10. "go-common/library/xstr"
  11. )
  12. var (
  13. _emptActDetail = make([]*model.ActiveDetail, 0)
  14. _emptActModule = make([]*model.Module, 0)
  15. _emptActVideos = make([]*arcmdl.Arc, 0)
  16. _emptTreeList = make([][]*model.TreeList, 0)
  17. )
  18. //ArcsInfo archive info
  19. func (s *Service) ArcsInfo(c context.Context, aids []int64) (arc []*arcmdl.Arc, err error) {
  20. var (
  21. arcsReply *arcmdl.ArcsReply
  22. res map[int64]*arcmdl.Arc
  23. )
  24. if arcsReply, err = s.arcClient.Arcs(c, &arcmdl.ArcsRequest{Aids: aids}); err != nil {
  25. log.Error("s.arcClient.Arcs(%v) error(%v)", aids, err)
  26. return
  27. }
  28. if arcsReply == nil {
  29. return
  30. }
  31. res = make(map[int64]*arcmdl.Arc)
  32. for k, v := range arcsReply.Arcs {
  33. if v != nil && v.IsNormal() {
  34. res[k] = v
  35. }
  36. }
  37. for _, aid := range aids {
  38. if v, ok := res[aid]; ok {
  39. arc = append(arc, v)
  40. }
  41. }
  42. return
  43. }
  44. // ActModules matchs active videos
  45. func (s *Service) ActModules(c context.Context, mmid int64) (res []*arcmdl.Arc, err error) {
  46. var (
  47. mModule *model.Module
  48. aids []int64
  49. )
  50. if res, err = s.dao.GetActModuleCache(c, mmid); err != nil || res == nil {
  51. if mModule, err = s.dao.Module(c, mmid); err != nil {
  52. return
  53. }
  54. if mModule == nil {
  55. err = ecode.EsportsActVideoNotExist
  56. return
  57. }
  58. if aids, err = xstr.SplitInts(mModule.Oids); err != nil {
  59. return
  60. }
  61. if res, err = s.ArcsInfo(c, aids); err != nil {
  62. return
  63. }
  64. if res == nil {
  65. res = _emptActVideos
  66. return
  67. }
  68. s.cache.Do(c, func(c context.Context) {
  69. s.dao.AddActModuleCache(c, mmid, res)
  70. })
  71. }
  72. return
  73. }
  74. //MatchAct match act
  75. func (s *Service) MatchAct(c context.Context, aid int64) (act *model.Active, err error) {
  76. if act, err = s.dao.GetMActCache(c, aid); err != nil || act == nil {
  77. if act, err = s.dao.Active(c, aid); err != nil {
  78. log.Error("s.dao.Active error(%v)", err)
  79. return
  80. }
  81. if act != nil {
  82. s.cache.Do(c, func(c context.Context) {
  83. s.dao.AddMActCache(c, aid, act)
  84. })
  85. }
  86. }
  87. return
  88. }
  89. // ActPage matchs active page info
  90. func (s *Service) ActPage(c context.Context, aid int64) (res *model.ActivePage, err error) {
  91. var (
  92. act *model.Active
  93. modules []*model.Module
  94. aids []int64
  95. actDetail []*model.ActiveDetail
  96. videos []*arcmdl.Arc
  97. moduleErr, actDetailError, actError error
  98. mapSeasons map[int64]*model.Season
  99. )
  100. if res, err = s.dao.GetActPageCache(c, aid); err != nil || res == nil {
  101. res = &model.ActivePage{}
  102. group, errCtx := errgroup.WithContext(c)
  103. group.Go(func() error {
  104. if act, actError = s.MatchAct(errCtx, aid); actError != nil {
  105. log.Error("s.dao.Active error(%v)", moduleErr)
  106. }
  107. return actError
  108. })
  109. group.Go(func() error {
  110. if modules, moduleErr = s.dao.Modules(errCtx, aid); moduleErr != nil {
  111. log.Error("s.dao.Modules error(%v)", moduleErr)
  112. return nil
  113. }
  114. if len(modules) > 0 && modules[0].Oids != "" {
  115. if aids, moduleErr = xstr.SplitInts(modules[0].Oids); moduleErr != nil {
  116. log.Error("s.ActPage.SplitInts oids error(%v) error(%v)", modules[0].Oids, moduleErr)
  117. return nil
  118. }
  119. if videos, moduleErr = s.ArcsInfo(c, aids); moduleErr != nil {
  120. log.Error("s.ActPage.ArcsInfo aids(%v) error(%v)", aids, moduleErr)
  121. }
  122. }
  123. return nil
  124. })
  125. group.Go(func() error {
  126. if actDetail, actDetailError = s.dao.ActDetail(errCtx, aid); actDetailError != nil {
  127. log.Error("s.dao.ActDetail error(%v)", actDetailError)
  128. }
  129. return nil
  130. })
  131. err = group.Wait()
  132. if err != nil {
  133. return
  134. }
  135. if act == nil {
  136. err = ecode.EsportsActNotExist
  137. return
  138. }
  139. res.Active = act
  140. if len(actDetail) == 0 {
  141. res.ActiveDetail = _emptActDetail
  142. } else {
  143. res.ActiveDetail = actDetail
  144. }
  145. if len(modules) == 0 {
  146. res.Modules = _emptActModule
  147. } else {
  148. res.Modules = modules
  149. }
  150. if len(videos) == 0 {
  151. res.Videos = _emptActVideos
  152. } else {
  153. res.Videos = videos
  154. }
  155. if mapSeasons, err = s.dao.EpSeasons(c, []int64{act.Sid}); err != nil {
  156. log.Error("s.dao.EpSeasons error(%v)", err)
  157. err = nil
  158. }
  159. if season, ok := mapSeasons[act.Sid]; ok {
  160. res.Season = season
  161. } else {
  162. res.Season = &model.Season{}
  163. }
  164. s.cache.Do(c, func(c context.Context) {
  165. s.dao.AddActPageCache(c, aid, res)
  166. })
  167. }
  168. return
  169. }
  170. //ContestCommon act match same deal logic
  171. func (s *Service) ContestCommon(c context.Context, mid int64, p *model.ParamContest) (rs []*model.Contest, total int, err error) {
  172. var (
  173. cData, tmpRs []*model.Contest
  174. contErr, teamErr, seasonErr error
  175. teams, seasons []*model.Filter
  176. cids []int64
  177. dbContests map[int64]*model.Contest
  178. )
  179. group, errCtx := errgroup.WithContext(c)
  180. group.Go(func() error {
  181. if cData, total, contErr = s.dao.SearchContestQuery(errCtx, p); contErr != nil {
  182. log.Error("s.dao.SearchContest error(%v)", contErr)
  183. }
  184. return contErr
  185. })
  186. group.Go(func() error {
  187. if teams, teamErr = s.dao.Teams(errCtx); teamErr != nil {
  188. log.Error("s.dao.Teams error(%v)", teamErr)
  189. }
  190. return nil
  191. })
  192. group.Go(func() error {
  193. if seasons, seasonErr = s.dao.SeasonAll(errCtx); seasonErr != nil {
  194. log.Error("s.dao.SeasonAll error %v", seasonErr)
  195. }
  196. return nil
  197. })
  198. err = group.Wait()
  199. if err != nil {
  200. return
  201. }
  202. if total == 0 || len(cData) == 0 {
  203. rs = _emptContest
  204. return
  205. }
  206. cids = s.contestIDs(cData)
  207. if len(cids) > 0 {
  208. if dbContests, err = s.dao.EpContests(c, cids); err != nil {
  209. log.Error("s.dao.Contest error(%v)", err)
  210. return
  211. }
  212. }
  213. for _, c := range cData {
  214. if contest, ok := dbContests[c.ID]; ok {
  215. tmpRs = append(tmpRs, contest)
  216. }
  217. }
  218. rs = s.ContestInfo(c, cids, tmpRs, teams, seasons, mid)
  219. return
  220. }
  221. // ActTop act match top data
  222. func (s *Service) ActTop(c context.Context, mid int64, param *model.ParamActTop) (res []*model.Contest, total int, err error) {
  223. var (
  224. act *model.Active
  225. sids []int64
  226. mapSeasons map[int64]*model.Season
  227. sTime, eTime time.Time
  228. season *model.Season
  229. ok bool
  230. )
  231. isFirst := param.Pn == 1 && param.Sort == 0 && param.Stime == ""
  232. if isFirst {
  233. if res, total, err = s.dao.GetActTopCache(c, param.Aid, int64(param.Ps)); err != nil {
  234. err = nil
  235. } else if len(res) > 0 {
  236. s.fmtContest(c, res, mid)
  237. return
  238. }
  239. }
  240. if act, err = s.MatchAct(c, param.Aid); err != nil {
  241. log.Error("s.MatchAct error(%v)", err)
  242. return
  243. }
  244. if act == nil || act.Sid <= 0 {
  245. err = ecode.EsportsActNotExist
  246. return
  247. }
  248. sids = []int64{act.Sid}
  249. if mapSeasons, err = s.dao.EpSeasons(c, sids); err != nil {
  250. log.Error("MatchAct s.dao.EpSeasons error(%v)", err)
  251. return
  252. }
  253. season, ok = mapSeasons[act.Sid]
  254. if !ok {
  255. log.Error("s.ActTop sid(%d) not exists", act.Sid)
  256. return
  257. }
  258. if param.Stime != "" {
  259. sTime, _ = time.ParseInLocation("2006-01-02", param.Stime, time.Local)
  260. if sTime.Unix() <= season.Stime {
  261. param.Stime = time.Unix(season.Stime, 0).Format("2006-01-02 15:04:05")
  262. } else {
  263. sTime, _ = time.ParseInLocation("2006-01-02", param.Stime, time.Local)
  264. param.Stime = time.Unix(sTime.Unix(), 0).Format("2006-01-02") + " 00:00:00"
  265. }
  266. } else {
  267. param.Stime = time.Unix(season.Stime, 0).Format("2006-01-02 15:04:05")
  268. }
  269. if param.Etime != "" {
  270. if season.Etime != 0 {
  271. eTime, _ = time.ParseInLocation("2006-01-02", param.Etime, time.Local)
  272. if eTime.Unix() >= season.Etime {
  273. param.Etime = time.Unix(season.Etime, 0).Format("2006-01-02 15:04:05")
  274. } else {
  275. eTime, _ = time.ParseInLocation("2006-01-02", param.Etime, time.Local)
  276. param.Etime = time.Unix(eTime.Unix(), 0).Format("2006-01-02") + " 23:59:59"
  277. }
  278. } else {
  279. eTime, _ = time.ParseInLocation("2006-01-02", param.Etime, time.Local)
  280. param.Etime = time.Unix(eTime.Unix(), 0).Format("2006-01-02") + " 23:59:59"
  281. }
  282. } else {
  283. if season.Etime != 0 {
  284. param.Etime = time.Unix(season.Etime, 0).Format("2006-01-02 15:04:05")
  285. }
  286. }
  287. p := &model.ParamContest{
  288. Mid: act.Mid,
  289. Sids: []int64{act.Sid},
  290. Sort: param.Sort,
  291. Stime: param.Stime,
  292. Etime: param.Etime,
  293. Pn: param.Pn,
  294. Ps: param.Ps,
  295. }
  296. if res, total, err = s.ContestCommon(c, mid, p); err != nil {
  297. return
  298. }
  299. if len(res) == 0 {
  300. res = _emptContest
  301. return
  302. }
  303. if isFirst {
  304. s.cache.Do(c, func(c context.Context) {
  305. s.dao.AddActTopCache(c, param.Aid, int64(param.Ps), res, total)
  306. })
  307. }
  308. return
  309. }
  310. // ActPoints act match point match
  311. func (s *Service) ActPoints(c context.Context, mid int64, param *model.ParamActPoint) (res []*model.Contest, total int, err error) {
  312. var (
  313. act *model.Active
  314. detail *model.ActiveDetail
  315. )
  316. isFirst := param.Pn == 1 && param.Sort == 0
  317. if isFirst {
  318. if res, total, err = s.dao.GetActPointsCache(c, param.Aid, param.MdID, int64(param.Ps)); err != nil {
  319. err = nil
  320. } else if len(res) > 0 {
  321. s.fmtContest(c, res, mid)
  322. return
  323. }
  324. }
  325. if act, err = s.MatchAct(c, param.Aid); err != nil {
  326. log.Error("s.dao.Active error(%v)", err)
  327. return
  328. }
  329. if act == nil || act.Sid <= 0 {
  330. err = ecode.EsportsActNotExist
  331. return
  332. }
  333. p := &model.ParamContest{
  334. Mid: act.Mid,
  335. Sids: []int64{act.Sid},
  336. Sort: param.Sort,
  337. Pn: param.Pn,
  338. Ps: param.Ps,
  339. }
  340. detail, err = s.dao.PActDetail(c, param.MdID)
  341. if err != nil {
  342. log.Error("s.dao.PActDetail error(%v)", err)
  343. return
  344. }
  345. if detail != nil {
  346. if detail.STime != 0 {
  347. p.Stime = time.Unix(detail.STime, 0).Format("2006-01-02 15:04:05")
  348. }
  349. if detail.ETime != 0 {
  350. p.Etime = time.Unix(detail.ETime, 0).Format("2006-01-02 15:04:05")
  351. }
  352. } else {
  353. err = ecode.EsportsActPointNotExist
  354. return
  355. }
  356. if res, total, err = s.ContestCommon(c, mid, p); err != nil {
  357. return
  358. }
  359. if len(res) == 0 {
  360. res = _emptContest
  361. return
  362. }
  363. if isFirst {
  364. s.cache.Do(c, func(c context.Context) {
  365. s.dao.AddActPointsCache(c, param.Aid, param.MdID, int64(param.Ps), res, total)
  366. })
  367. }
  368. return
  369. }
  370. //contestTeam get contest team
  371. func (s *Service) contestTeam(c context.Context, contests map[int64]*model.Contest, teams map[int64]*model.Filter) (list []*model.ContestInfo, err error) {
  372. for _, v := range contests {
  373. contest := &model.ContestInfo{Contest: v}
  374. if v.HomeID > 0 {
  375. if team, ok := teams[v.HomeID]; ok {
  376. contest.HomeName = team.Title
  377. contest.HomeTeam = team
  378. } else {
  379. contest.HomeName = ""
  380. contest.HomeTeam = struct{}{}
  381. }
  382. } else {
  383. contest.HomeName = ""
  384. contest.HomeTeam = struct{}{}
  385. }
  386. if v.AwayID > 0 {
  387. if team, ok := teams[v.AwayID]; ok {
  388. contest.AwayName = team.Title
  389. contest.AwayTeam = team
  390. } else {
  391. contest.AwayName = ""
  392. contest.AwayTeam = struct{}{}
  393. }
  394. } else {
  395. contest.AwayName = ""
  396. contest.AwayTeam = struct{}{}
  397. }
  398. if v.SuccessTeam > 0 {
  399. if team, ok := teams[v.SuccessTeam]; ok {
  400. contest.SuccessName = team.Title
  401. } else {
  402. contest.SuccessName = ""
  403. }
  404. } else {
  405. contest.SuccessName = ""
  406. }
  407. contest.Season = struct{}{}
  408. contest.SuccessTeaminfo = struct{}{}
  409. list = append(list, contest)
  410. }
  411. return
  412. }
  413. //TeamMap get team and map team id to team
  414. func (s *Service) TeamMap(ctx context.Context) (res map[int64]*model.Filter, err error) {
  415. var (
  416. teams []*model.Filter
  417. )
  418. if teams, err = s.dao.Teams(ctx); err != nil {
  419. log.Error("LoadKnockoutTree s.dao.Teams error(%v)", err)
  420. return
  421. }
  422. res = make(map[int64]*model.Filter)
  423. for _, v := range teams {
  424. res[v.ID] = v
  425. }
  426. return
  427. }
  428. // BuildKnockTree knock tree.
  429. func (s *Service) BuildKnockTree(c context.Context) {
  430. var (
  431. trees []*model.Tree
  432. sTree []*model.TreeList
  433. mids []int64
  434. contests map[int64]*model.Contest
  435. cInfos []*model.ContestInfo
  436. mapContests map[int64]*model.ContestInfo
  437. treeList [][]*model.TreeList
  438. err error
  439. details []*model.ActiveDetail
  440. teamMap map[int64]*model.Filter
  441. )
  442. if teamMap, err = s.TeamMap(c); err != nil {
  443. log.Error("LoadKnockoutTree TeamMap error(%v)", err)
  444. return
  445. }
  446. if details, err = s.dao.KDetails(c); err != nil {
  447. log.Error("LoadKnockoutTree s.dao.KDetails error(%v)", err)
  448. return
  449. }
  450. //build tree
  451. for _, detail := range details {
  452. if detail.Online == _downline {
  453. //edit status; must add cache time
  454. if s.dao.AddActKnockCacheTime(c, detail.ID); err != nil {
  455. log.Error("LoadKnockoutTree s.dao.AddActKnockCacheTime error(%v)", err)
  456. continue
  457. }
  458. } else {
  459. treeList = _emptTreeList
  460. sTree = make([]*model.TreeList, 0)
  461. if trees, err = s.dao.Trees(c, detail.ID); err != nil {
  462. log.Error("s.dao.Trees error(%v)", err)
  463. return
  464. }
  465. for _, tree := range trees {
  466. mids = append(mids, tree.Mid)
  467. }
  468. if len(mids) == 0 {
  469. continue
  470. } else {
  471. count := len(mids)
  472. if contests, err = s.dao.EpContests(c, mids); err != nil {
  473. log.Error("s.dao.RawEpContests error(%v)", err)
  474. continue
  475. }
  476. if cInfos, err = s.contestTeam(c, contests, teamMap); err != nil {
  477. log.Error("s.contestTeam Error (%v)", err)
  478. continue
  479. }
  480. mapContests = make(map[int64]*model.ContestInfo, count)
  481. for _, info := range cInfos {
  482. mapContests[info.ID] = info
  483. }
  484. for _, tree := range trees {
  485. if tree.Pid == 0 {
  486. if len(sTree) > 0 {
  487. treeList = append(treeList, sTree)
  488. }
  489. sTree = nil
  490. if cInfo, ok := mapContests[tree.Mid]; ok {
  491. sTree = append(sTree, &model.TreeList{Tree: tree, ContestInfo: cInfo})
  492. } else {
  493. sTree = append(sTree, &model.TreeList{Tree: tree})
  494. }
  495. } else {
  496. if cInfo, ok := mapContests[tree.Mid]; ok {
  497. sTree = append(sTree, &model.TreeList{Tree: tree, ContestInfo: cInfo})
  498. } else {
  499. sTree = append(sTree, &model.TreeList{Tree: tree})
  500. }
  501. }
  502. }
  503. if len(sTree) > 0 {
  504. treeList = append(treeList, sTree)
  505. }
  506. if len(treeList) > 0 {
  507. go s.dao.AddActKnockoutCache(context.Background(), detail.ID, treeList)
  508. }
  509. }
  510. }
  511. }
  512. }
  513. // ActKnockout knockout tree
  514. func (s *Service) ActKnockout(c context.Context, madID int64) (res [][]*model.TreeList, err error) {
  515. if res, err = s.dao.GetActKnockoutCache(c, madID); err != nil {
  516. return
  517. }
  518. if len(res) == 0 {
  519. res = _emptTreeList
  520. err = ecode.EsportsActKnockNotExist
  521. }
  522. return
  523. }