match.go 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816
  1. package service
  2. import (
  3. "context"
  4. "sort"
  5. "strconv"
  6. "time"
  7. "go-common/app/interface/main/esports/model"
  8. arcmdl "go-common/app/service/main/archive/api"
  9. "go-common/library/ecode"
  10. "go-common/library/log"
  11. "go-common/library/sync/errgroup"
  12. )
  13. const (
  14. _gameNoSub = 6
  15. _gameSub = 3
  16. _gameIn = 5
  17. _gameLive = 4
  18. _gameOver = 1
  19. _caleDay = 3
  20. _typeMatch = "matchs"
  21. _typeGame = "games"
  22. _typeTeam = "teams"
  23. _typeYear = "years"
  24. _typeTag = "tags"
  25. _downline = 0
  26. )
  27. var (
  28. _emptContest = make([]*model.Contest, 0)
  29. _emptCalendar = make([]*model.Calendar, 0)
  30. _emptFilter = make([]*model.Filter, 0)
  31. _emptVideoList = make([]*arcmdl.Arc, 0)
  32. _emptSeason = make([]*model.Season, 0)
  33. _emptContestDetail = make([]*model.ContestsData, 0)
  34. )
  35. // FilterMatch filter match.
  36. func (s *Service) FilterMatch(c context.Context, p *model.ParamFilter) (rs map[string][]*model.Filter, err error) {
  37. var (
  38. tmpRs map[string][]*model.Filter
  39. fm *model.FilterES
  40. fMap map[string]map[int64]*model.Filter
  41. matchs, games, teams []*model.Filter
  42. )
  43. isAll := p.Tid == 0 && p.Gid == 0 && p.Mid == 0 && p.Stime == ""
  44. if rs, err = s.dao.FMatCache(c); err != nil {
  45. err = nil
  46. }
  47. if isAll && len(rs) > 0 {
  48. return
  49. }
  50. matchs, games, teams = s.filterLeft(c)
  51. tmpRs = make(map[string][]*model.Filter, 3)
  52. tmpRs[_typeMatch] = matchs
  53. tmpRs[_typeGame] = games
  54. tmpRs[_typeTeam] = teams
  55. if fm, err = s.dao.FilterMatch(c, p); err != nil {
  56. log.Error("s.dao.FilterMatch error(%v)", err)
  57. return
  58. }
  59. fMap = s.filterMap(tmpRs)
  60. if tmpRs, err = s.fmtES(fm, fMap); err != nil {
  61. log.Error("FilterMatch s.filterES error(%v)", err)
  62. }
  63. rs = make(map[string][]*model.Filter, 3)
  64. rs[_typeMatch] = tmpRs[_typeMatch]
  65. rs[_typeGame] = tmpRs[_typeGame]
  66. rs[_typeTeam] = tmpRs[_typeTeam]
  67. if isAll {
  68. s.cache.Do(c, func(c context.Context) {
  69. s.dao.SetFMatCache(c, rs)
  70. })
  71. } else {
  72. if len(rs[_typeMatch]) == 0 && len(rs[_typeGame]) == 0 && len(rs[_typeTeam]) == 0 {
  73. if tmpRs, err = s.dao.FMatCache(c); err != nil {
  74. err = nil
  75. }
  76. if len(tmpRs) > 0 {
  77. rs = tmpRs
  78. }
  79. }
  80. }
  81. return
  82. }
  83. func (s *Service) filterLeft(c context.Context) (matchs, games, teams []*model.Filter) {
  84. var (
  85. matchErr, gameErr, teamErr error
  86. )
  87. group := &errgroup.Group{}
  88. group.Go(func() error {
  89. if matchs, matchErr = s.dao.Matchs(context.Background()); matchErr != nil {
  90. log.Error("s.dao.Matchs error %v", matchErr)
  91. }
  92. return nil
  93. })
  94. group.Go(func() error {
  95. if games, gameErr = s.dao.Games(context.Background()); gameErr != nil {
  96. log.Error("s.dao.Games error %v", gameErr)
  97. }
  98. return nil
  99. })
  100. group.Go(func() error {
  101. if teams, teamErr = s.dao.Teams(context.Background()); teamErr != nil {
  102. log.Error("s.dao.Teams error %v", teamErr)
  103. }
  104. return nil
  105. })
  106. group.Wait()
  107. if len(matchs) == 0 {
  108. matchs = _emptFilter
  109. }
  110. if len(games) == 0 {
  111. games = _emptFilter
  112. }
  113. if len(teams) == 0 {
  114. teams = _emptFilter
  115. }
  116. return
  117. }
  118. // Calendar contest calendar count
  119. func (s *Service) Calendar(c context.Context, p *model.ParamFilter) (rs []*model.Calendar, err error) {
  120. var fc map[string]int64
  121. before3 := time.Now().AddDate(0, 0, -_caleDay).Format("2006-01-02")
  122. after3 := time.Now().AddDate(0, 0, _caleDay).Format("2006-01-02")
  123. todayAll := p.Mid == 0 && p.Gid == 0 && p.Tid == 0 && p.Stime == before3 && p.Etime == after3
  124. if todayAll {
  125. if rs, err = s.dao.CalendarCache(c, p); err != nil {
  126. err = nil
  127. }
  128. if len(rs) > 0 {
  129. return
  130. }
  131. }
  132. if fc, err = s.dao.FilterCale(c, p); err != nil {
  133. log.Error("s.dao.FilterCale error(%v)", err)
  134. return
  135. }
  136. if len(fc) == 0 {
  137. rs = _emptCalendar
  138. return
  139. }
  140. for d, c := range fc {
  141. rs = append(rs, &model.Calendar{Stime: d, Count: c})
  142. }
  143. if todayAll {
  144. s.cache.Do(c, func(c context.Context) {
  145. s.dao.SetCalendarCache(c, p, rs)
  146. })
  147. }
  148. return
  149. }
  150. func (s *Service) fmtContest(c context.Context, contests []*model.Contest, mid int64) {
  151. cids := s.contestIDs(contests)
  152. favContest, _ := s.isFavs(c, mid, cids)
  153. for _, contest := range contests {
  154. if contest.Etime > 0 && time.Now().Unix() > contest.Etime {
  155. contest.GameState = _gameOver
  156. } else if contest.Stime <= time.Now().Unix() && (contest.Etime >= time.Now().Unix() || contest.Etime == 0) {
  157. if contest.LiveRoom == 0 {
  158. contest.GameState = _gameIn
  159. } else {
  160. contest.GameState = _gameLive
  161. }
  162. } else if contest.LiveRoom > 0 {
  163. if v, ok := favContest[contest.ID]; ok && v && mid > 0 {
  164. contest.GameState = _gameSub
  165. } else {
  166. contest.GameState = _gameNoSub
  167. }
  168. }
  169. }
  170. }
  171. // ListContest contest list.
  172. func (s *Service) ListContest(c context.Context, mid int64, p *model.ParamContest) (rs []*model.Contest, total int, err error) {
  173. var (
  174. teams, seasons []*model.Filter
  175. cData, tmpRs []*model.Contest
  176. dbContests map[int64]*model.Contest
  177. group *errgroup.Group
  178. cids []int64
  179. contErr, teamErr, seasonErr error
  180. )
  181. // get from cache.
  182. isFirst := p.Mid == 0 && p.Gid == 0 && p.Tid == 0 && p.Stime == "" && p.GState == "" && p.Pn == 1 && len(p.Sids) == 0 && p.Sort == 0
  183. if isFirst {
  184. if rs, total, err = s.dao.ContestCache(c, p.Ps); err != nil {
  185. err = nil
  186. } else if len(rs) > 0 {
  187. s.fmtContest(c, rs, mid)
  188. return
  189. }
  190. }
  191. group, errCtx := errgroup.WithContext(c)
  192. group.Go(func() error {
  193. if cData, total, contErr = s.dao.SearchContest(errCtx, p); contErr != nil {
  194. log.Error("s.dao.SearchContest error(%v)", contErr)
  195. }
  196. return contErr
  197. })
  198. group.Go(func() error {
  199. if teams, teamErr = s.dao.Teams(errCtx); teamErr != nil {
  200. log.Error("s.dao.Teams error(%v)", teamErr)
  201. }
  202. return nil
  203. })
  204. group.Go(func() error {
  205. if seasons, seasonErr = s.dao.SeasonAll(errCtx); seasonErr != nil {
  206. log.Error("s.dao.SeasonAll error %v", seasonErr)
  207. }
  208. return nil
  209. })
  210. err = group.Wait()
  211. if err != nil {
  212. return
  213. }
  214. if total == 0 || len(cData) == 0 {
  215. rs = _emptContest
  216. return
  217. }
  218. cids = s.contestIDs(cData)
  219. if len(cids) > 0 {
  220. if dbContests, err = s.dao.EpContests(c, cids); err != nil {
  221. log.Error("s.dao.EpContests error(%v)", err)
  222. return
  223. }
  224. }
  225. for _, c := range cData {
  226. if contest, ok := dbContests[c.ID]; ok {
  227. tmpRs = append(tmpRs, contest)
  228. }
  229. }
  230. rs = s.ContestInfo(c, cids, tmpRs, teams, seasons, mid)
  231. if isFirst {
  232. s.cache.Do(c, func(c context.Context) {
  233. s.dao.SetContestCache(c, p.Ps, rs, total)
  234. })
  235. }
  236. return
  237. }
  238. func (s *Service) contestIDs(cData []*model.Contest) (rs []int64) {
  239. for _, contest := range cData {
  240. rs = append(rs, contest.ID)
  241. }
  242. return
  243. }
  244. // ContestInfo contest add team season.
  245. func (s *Service) ContestInfo(c context.Context, cids []int64, cData []*model.Contest, teams, seasons []*model.Filter, mid int64) (rs []*model.Contest) {
  246. var (
  247. mapTeam, mapSeason map[int64]*model.Filter
  248. )
  249. mapTeam = make(map[int64]*model.Filter, len(teams))
  250. for _, team := range teams {
  251. mapTeam[team.ID] = team
  252. }
  253. mapSeason = make(map[int64]*model.Filter, len(seasons))
  254. for _, season := range seasons {
  255. mapSeason[season.ID] = season
  256. }
  257. favContest, _ := s.isFavs(c, mid, cids)
  258. for _, contest := range cData {
  259. if contest == nil {
  260. continue
  261. }
  262. if v, ok := mapTeam[contest.HomeID]; ok && v != nil {
  263. contest.HomeTeam = v
  264. } else {
  265. contest.HomeTeam = struct{}{}
  266. }
  267. if v, ok := mapTeam[contest.AwayID]; ok && v != nil {
  268. contest.AwayTeam = v
  269. } else {
  270. contest.AwayTeam = struct{}{}
  271. }
  272. if v, ok := mapTeam[contest.SuccessTeam]; ok && v != nil {
  273. contest.SuccessTeaminfo = v
  274. } else {
  275. contest.SuccessTeaminfo = struct{}{}
  276. }
  277. if v, ok := mapSeason[contest.Sid]; ok && v != nil {
  278. contest.Season = v
  279. } else {
  280. contest.Season = struct{}{}
  281. }
  282. if contest.Etime > 0 && time.Now().Unix() > contest.Etime {
  283. contest.GameState = _gameOver
  284. } else if contest.Stime <= time.Now().Unix() && (contest.Etime >= time.Now().Unix() || contest.Etime == 0) {
  285. if contest.LiveRoom == 0 {
  286. contest.GameState = _gameIn
  287. } else {
  288. contest.GameState = _gameLive
  289. }
  290. } else if contest.LiveRoom > 0 {
  291. if v, ok := favContest[contest.ID]; ok && v && mid > 0 {
  292. contest.GameState = _gameSub
  293. } else {
  294. contest.GameState = _gameNoSub
  295. }
  296. }
  297. rs = append(rs, contest)
  298. }
  299. return
  300. }
  301. // ListVideo video list.
  302. func (s *Service) ListVideo(c context.Context, p *model.ParamVideo) (rs []*arcmdl.Arc, total int, err error) {
  303. var (
  304. vData []*model.SearchVideo
  305. aids []int64
  306. arcsReply *arcmdl.ArcsReply
  307. )
  308. isFirst := p.Mid == 0 && p.Gid == 0 && p.Tid == 0 && p.Year == 0 && p.Tag == 0 && p.Sort == 0 && p.Pn == 1
  309. if isFirst {
  310. // get from cache.
  311. if rs, total, err = s.dao.VideoCache(c, p.Ps); err != nil {
  312. err = nil
  313. } else if len(rs) > 0 {
  314. return
  315. }
  316. }
  317. if vData, total, err = s.dao.SearchVideo(c, p); err != nil {
  318. log.Error("s.dao.SearchVideo(%v) error(%v)", p, err)
  319. return
  320. }
  321. if total == 0 {
  322. rs = _emptVideoList
  323. return
  324. }
  325. for _, arc := range vData {
  326. aids = append(aids, arc.AID)
  327. }
  328. if arcsReply, err = s.arcClient.Arcs(c, &arcmdl.ArcsRequest{Aids: aids}); err != nil {
  329. log.Error("ListVideo s.arc.Archives3 error(%v)", err)
  330. return
  331. }
  332. for _, aid := range aids {
  333. if arc, ok := arcsReply.Arcs[aid]; ok && arc.IsNormal() {
  334. rs = append(rs, arc)
  335. }
  336. }
  337. if isFirst {
  338. s.cache.Do(c, func(c context.Context) {
  339. s.dao.SetVideoCache(c, p.Ps, rs, total)
  340. })
  341. }
  342. return
  343. }
  344. // FilterVideo filter video.
  345. func (s *Service) FilterVideo(c context.Context, p *model.ParamFilter) (rs map[string][]*model.Filter, err error) {
  346. var (
  347. tmpRs map[string][]*model.Filter
  348. fv *model.FilterES
  349. fMap map[string]map[int64]*model.Filter
  350. matchs, games, teams, tags, years []*model.Filter
  351. )
  352. isAll := p.Year == 0 && p.Tag == 0 && p.Tid == 0 && p.Gid == 0 && p.Mid == 0
  353. if rs, err = s.dao.FVideoCache(c); err != nil {
  354. err = nil
  355. }
  356. if isAll && len(rs) > 0 {
  357. return
  358. }
  359. matchs, games, teams, tags, years = s.filterTop(c)
  360. tmpRs = make(map[string][]*model.Filter, 3)
  361. tmpRs[_typeMatch] = matchs
  362. tmpRs[_typeGame] = games
  363. tmpRs[_typeTeam] = teams
  364. tmpRs[_typeYear] = years
  365. tmpRs[_typeTag] = tags
  366. if fv, err = s.dao.FilterVideo(c, p); err != nil {
  367. log.Error("s.dao.FilterVideo error(%v)", err)
  368. return
  369. }
  370. fMap = s.filterMap(tmpRs)
  371. if rs, err = s.fmtES(fv, fMap); err != nil {
  372. log.Error("FilterVideo s.filterES error(%v)", err)
  373. }
  374. if isAll {
  375. s.cache.Do(c, func(c context.Context) {
  376. s.dao.SetFVideoCache(c, rs)
  377. })
  378. } else {
  379. if len(rs[_typeMatch]) == 0 && len(rs[_typeGame]) == 0 && len(rs[_typeTeam]) == 0 && len(rs[_typeYear]) == 0 && len(rs[_typeTag]) == 0 {
  380. if tmpRs, err = s.dao.FVideoCache(c); err != nil {
  381. err = nil
  382. }
  383. if len(tmpRs) > 0 {
  384. rs = tmpRs
  385. }
  386. }
  387. }
  388. return
  389. }
  390. func (s *Service) filterTop(c context.Context) (matchs, games, teams, tags, years []*model.Filter) {
  391. var (
  392. matchErr, gameErr, teamErr, tagErr, yearErr error
  393. )
  394. group := &errgroup.Group{}
  395. group.Go(func() error {
  396. if matchs, matchErr = s.dao.Matchs(context.Background()); matchErr != nil {
  397. log.Error("s.dao.Matchs error %v", matchErr)
  398. }
  399. return nil
  400. })
  401. group.Go(func() error {
  402. if games, gameErr = s.dao.Games(context.Background()); gameErr != nil {
  403. log.Error("s.dao.Games error %v", gameErr)
  404. }
  405. return nil
  406. })
  407. group.Go(func() error {
  408. if teams, teamErr = s.dao.Teams(context.Background()); teamErr != nil {
  409. log.Error("s.dao.Teams error %v", teamErr)
  410. }
  411. return nil
  412. })
  413. group.Go(func() error {
  414. if tags, tagErr = s.dao.Tags(context.Background()); tagErr != nil {
  415. log.Error("s.dao.tags error %v", tagErr)
  416. }
  417. return nil
  418. })
  419. group.Go(func() error {
  420. if years, yearErr = s.dao.Years(context.Background()); yearErr != nil {
  421. log.Error("s.dao.Years error %v", yearErr)
  422. }
  423. return nil
  424. })
  425. group.Wait()
  426. if len(matchs) == 0 {
  427. matchs = _emptFilter
  428. }
  429. if len(games) == 0 {
  430. games = _emptFilter
  431. }
  432. if len(teams) == 0 {
  433. teams = _emptFilter
  434. }
  435. if len(years) == 0 {
  436. years = _emptFilter
  437. }
  438. if len(tags) == 0 {
  439. tags = _emptFilter
  440. }
  441. return
  442. }
  443. func (s *Service) fmtES(fv *model.FilterES, fMap map[string]map[int64]*model.Filter) (rs map[string][]*model.Filter, err error) {
  444. var (
  445. intMid, intGid, intTeam, intTag, intYear int64
  446. matchs, games, teams, tags, years []*model.Filter
  447. )
  448. group := &errgroup.Group{}
  449. group.Go(func() error {
  450. for _, midGroup := range fv.GroupByMatch {
  451. if intMid, err = strconv.ParseInt(midGroup.Key, 10, 64); err != nil {
  452. err = nil
  453. continue
  454. }
  455. if match, ok := fMap[_typeMatch][intMid]; ok {
  456. matchs = append(matchs, match)
  457. }
  458. }
  459. return nil
  460. })
  461. group.Go(func() error {
  462. for _, gidGroup := range fv.GroupByGid {
  463. if intGid, err = strconv.ParseInt(gidGroup.Key, 10, 64); err != nil {
  464. err = nil
  465. continue
  466. }
  467. if game, ok := fMap[_typeGame][intGid]; ok {
  468. games = append(games, game)
  469. }
  470. }
  471. return nil
  472. })
  473. group.Go(func() error {
  474. for _, teamGroup := range fv.GroupByTeam {
  475. if intTeam, err = strconv.ParseInt(teamGroup.Key, 10, 64); err != nil {
  476. err = nil
  477. continue
  478. }
  479. if team, ok := fMap[_typeTeam][intTeam]; ok {
  480. teams = append(teams, team)
  481. }
  482. }
  483. return nil
  484. })
  485. group.Go(func() error {
  486. for _, tagGroup := range fv.GroupByTag {
  487. if intTag, err = strconv.ParseInt(tagGroup.Key, 10, 64); err != nil {
  488. err = nil
  489. continue
  490. }
  491. if tag, ok := fMap[_typeTag][intTag]; ok {
  492. tags = append(tags, tag)
  493. }
  494. }
  495. return nil
  496. })
  497. group.Go(func() error {
  498. for _, yearGroup := range fv.GroupByYear {
  499. if intYear, err = strconv.ParseInt(yearGroup.Key, 10, 64); err != nil {
  500. err = nil
  501. continue
  502. }
  503. if year, ok := fMap[_typeYear][intYear]; ok {
  504. years = append(years, year)
  505. }
  506. }
  507. return nil
  508. })
  509. group.Wait()
  510. rs = make(map[string][]*model.Filter, 5)
  511. if len(matchs) == 0 {
  512. matchs = _emptFilter
  513. } else {
  514. sort.Slice(matchs, func(i, j int) bool {
  515. return matchs[i].Rank > matchs[j].Rank || (matchs[i].Rank == matchs[j].Rank && matchs[i].ID < matchs[j].ID)
  516. })
  517. }
  518. if len(games) == 0 {
  519. games = _emptFilter
  520. } else {
  521. sort.Slice(games, func(i, j int) bool { return games[i].ID < games[j].ID })
  522. }
  523. if len(teams) == 0 {
  524. teams = _emptFilter
  525. } else {
  526. sort.Slice(teams, func(i, j int) bool { return teams[i].ID < teams[j].ID })
  527. }
  528. if len(years) == 0 {
  529. years = _emptFilter
  530. } else {
  531. sort.Slice(years, func(i, j int) bool { return years[i].ID < years[j].ID })
  532. }
  533. if len(tags) == 0 {
  534. tags = _emptFilter
  535. } else {
  536. sort.Slice(tags, func(i, j int) bool { return tags[i].ID < tags[j].ID })
  537. }
  538. rs[_typeMatch] = matchs
  539. rs[_typeGame] = games
  540. rs[_typeTeam] = teams
  541. rs[_typeTag] = tags
  542. rs[_typeYear] = years
  543. return
  544. }
  545. func (s *Service) filterMap(f map[string][]*model.Filter) (rs map[string]map[int64]*model.Filter) {
  546. var (
  547. match, game, team, tag, year *model.Filter
  548. mapMatch, mapGame, mapTeam, mapTag, mapYear map[int64]*model.Filter
  549. )
  550. group := &errgroup.Group{}
  551. group.Go(func() error {
  552. mapMatch = make(map[int64]*model.Filter, len(f[_typeMatch]))
  553. for _, match = range f[_typeMatch] {
  554. if match != nil {
  555. mapMatch[match.ID] = match
  556. }
  557. }
  558. return nil
  559. })
  560. group.Go(func() error {
  561. mapGame = make(map[int64]*model.Filter, len(f[_typeGame]))
  562. for _, game = range f[_typeGame] {
  563. if game != nil {
  564. mapGame[game.ID] = game
  565. }
  566. }
  567. return nil
  568. })
  569. group.Go(func() error {
  570. mapTeam = make(map[int64]*model.Filter, len(f[_typeTeam]))
  571. for _, team = range f[_typeTeam] {
  572. if team != nil {
  573. mapTeam[team.ID] = team
  574. }
  575. }
  576. return nil
  577. })
  578. group.Go(func() error {
  579. mapTag = make(map[int64]*model.Filter, len(f[_typeTag]))
  580. for _, tag = range f[_typeTag] {
  581. if tag != nil {
  582. mapTag[tag.ID] = tag
  583. }
  584. }
  585. return nil
  586. })
  587. group.Go(func() error {
  588. mapYear = make(map[int64]*model.Filter, len(f[_typeYear]))
  589. for _, year = range f[_typeYear] {
  590. if year != nil {
  591. mapYear[year.ID] = year
  592. }
  593. }
  594. return nil
  595. })
  596. group.Wait()
  597. rs = make(map[string]map[int64]*model.Filter, 5)
  598. rs[_typeMatch] = mapMatch
  599. rs[_typeGame] = mapGame
  600. rs[_typeTeam] = mapTeam
  601. rs[_typeTag] = mapTag
  602. rs[_typeYear] = mapYear
  603. return
  604. }
  605. // Season season list.
  606. func (s *Service) Season(c context.Context, p *model.ParamSeason) (rs []*model.Season, count int, err error) {
  607. var (
  608. seasons []*model.Season
  609. start = (p.Pn - 1) * p.Ps
  610. end = start + p.Ps - 1
  611. )
  612. if rs, count, err = s.dao.SeasonCache(c, start, end); err != nil || len(rs) == 0 {
  613. err = nil
  614. if seasons, err = s.dao.Season(c); err != nil {
  615. log.Error("s.dao.Season error(%v)", err)
  616. return
  617. }
  618. count = len(seasons)
  619. if count == 0 || count < start {
  620. rs = _emptSeason
  621. return
  622. }
  623. s.cache.Do(c, func(c context.Context) {
  624. s.dao.SetSeasonCache(c, seasons, count)
  625. })
  626. if count > end+1 {
  627. rs = seasons[start : end+1]
  628. } else {
  629. rs = seasons[start:]
  630. }
  631. }
  632. return
  633. }
  634. // AppSeason app season list.
  635. func (s *Service) AppSeason(c context.Context, p *model.ParamSeason) (rs []*model.Season, count int, err error) {
  636. var (
  637. seasons []*model.Season
  638. start = (p.Pn - 1) * p.Ps
  639. end = start + p.Ps - 1
  640. )
  641. if rs, count, err = s.dao.SeasonMCache(c, start, end); err != nil || len(rs) == 0 {
  642. err = nil
  643. if seasons, err = s.dao.AppSeason(c); err != nil {
  644. log.Error("s.dao.AppSeason error(%v)", err)
  645. return
  646. }
  647. count = len(seasons)
  648. if count == 0 || count < start {
  649. rs = _emptSeason
  650. return
  651. }
  652. s.cache.Do(c, func(c context.Context) {
  653. s.dao.SetSeasonMCache(c, seasons, count)
  654. })
  655. if count > end+1 {
  656. rs = seasons[start : end+1]
  657. } else {
  658. rs = seasons[start:]
  659. }
  660. }
  661. sort.Slice(rs, func(i, j int) bool {
  662. return rs[i].Rank > rs[j].Rank || (rs[i].Rank == rs[j].Rank && rs[i].Stime > rs[j].Stime)
  663. })
  664. return
  665. }
  666. // Contest contest data.
  667. func (s *Service) Contest(c context.Context, mid, cid int64) (res *model.ContestDataPage, err error) {
  668. var (
  669. contest *model.Contest
  670. contestData []*model.ContestsData
  671. teams map[int64]*model.Team
  672. season map[int64]*model.Season
  673. teamErr, contestErr error
  674. games []*model.Game
  675. gameMap map[int64]*model.Game
  676. )
  677. if res, err = s.dao.GetCSingleData(c, cid); err != nil || res == nil {
  678. err = nil
  679. res = &model.ContestDataPage{}
  680. group, errCtx := errgroup.WithContext(c)
  681. group.Go(func() error {
  682. if contest, contestErr = s.dao.Contest(errCtx, cid); contestErr != nil {
  683. log.Error("SingleData.dao.Contest error(%v)", teamErr)
  684. }
  685. return contestErr
  686. })
  687. group.Go(func() error {
  688. if contestData, _ = s.dao.ContestData(errCtx, cid); err != nil {
  689. log.Error("SingleData.dao.ContestData error(%v)", teamErr)
  690. }
  691. return nil
  692. })
  693. err = group.Wait()
  694. if err != nil {
  695. return
  696. }
  697. if contest.ID == 0 {
  698. err = ecode.NothingFound
  699. return
  700. }
  701. if len(contestData) == 0 {
  702. contestData = _emptContestDetail
  703. }
  704. if teams, err = s.dao.EpTeams(c, []int64{contest.HomeID, contest.AwayID}); err != nil {
  705. log.Error("SingleData.dao.Teams error(%v)", err)
  706. err = nil
  707. }
  708. if season, err = s.dao.EpSeasons(c, []int64{contest.Sid}); err != nil {
  709. log.Error("SingleData.dao.EpSeasons error(%v)", err)
  710. err = nil
  711. }
  712. s.ContestInfos(contest, teams, season)
  713. res.Contest = contest
  714. res.Detail = contestData
  715. s.cache.Do(c, func(c context.Context) {
  716. s.dao.AddCSingleData(c, cid, res)
  717. })
  718. }
  719. if res.Contest.DataType == _lolType {
  720. games = s.lolGameMap.Data[res.Contest.MatchID]
  721. } else if res.Contest.DataType == _dotaType {
  722. games = s.dotaGameMap.Data[res.Contest.MatchID]
  723. }
  724. if len(games) > 0 {
  725. gameMap = make(map[int64]*model.Game, len(games))
  726. for _, game := range games {
  727. gameMap[game.ID] = game
  728. }
  729. for _, data := range res.Detail {
  730. if g, ok := gameMap[data.PointData]; ok {
  731. if g.Finished == true || (res.Contest.Etime > 0 && time.Now().Unix() > res.Contest.Etime) {
  732. data.GameStatus = 1
  733. } else if g.Finished == false {
  734. data.GameStatus = 2
  735. }
  736. }
  737. }
  738. }
  739. tmp := []*model.Contest{res.Contest}
  740. s.fmtContest(c, tmp, mid)
  741. return
  742. }
  743. // ContestInfos contest infos.
  744. func (s *Service) ContestInfos(contest *model.Contest, teams map[int64]*model.Team, season map[int64]*model.Season) {
  745. if homeTeam, ok := teams[contest.HomeID]; ok {
  746. contest.HomeTeam = homeTeam
  747. } else {
  748. contest.HomeTeam = struct{}{}
  749. }
  750. if awayTeam, ok := teams[contest.AwayID]; ok {
  751. contest.AwayTeam = awayTeam
  752. } else {
  753. contest.AwayTeam = struct{}{}
  754. }
  755. if sea, ok := season[contest.Sid]; ok {
  756. contest.Season = sea
  757. } else {
  758. contest.Season = struct{}{}
  759. }
  760. }
  761. // Recent contest recents.
  762. func (s *Service) Recent(c context.Context, mid int64, param *model.ParamCDRecent) (res []*model.Contest, err error) {
  763. var (
  764. teams map[int64]*model.Team
  765. season map[int64]*model.Season
  766. )
  767. if res, err = s.dao.GetCRecent(c, param); err != nil || len(res) == 0 {
  768. err = nil
  769. if res, err = s.dao.ContestRecent(c, param.HomeID, param.AwayID, param.CID, param.Ps); err != nil {
  770. log.Error("ContestRecent.dao.ContestRecent error(%v)", err)
  771. return
  772. }
  773. if len(res) == 0 {
  774. res = _emptContest
  775. return
  776. }
  777. for _, contest := range res {
  778. if teams, err = s.dao.EpTeams(c, []int64{contest.HomeID, contest.AwayID}); err != nil {
  779. log.Error("SingleData.dao.Teams error(%v)", err)
  780. err = nil
  781. }
  782. if season, err = s.dao.EpSeasons(c, []int64{contest.Sid}); err != nil {
  783. log.Error("SingleData.dao.EpSeasons error(%v)", err)
  784. err = nil
  785. }
  786. s.ContestInfos(contest, teams, season)
  787. contest.SuccessTeaminfo = struct{}{}
  788. }
  789. s.cache.Do(c, func(c context.Context) {
  790. s.dao.AddCRecent(c, param, res)
  791. })
  792. }
  793. s.fmtContest(c, res, mid)
  794. return
  795. }