rank.go 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. package rank
  2. import (
  3. "context"
  4. "fmt"
  5. "time"
  6. "go-common/app/interface/main/app-show/conf"
  7. accdao "go-common/app/interface/main/app-show/dao/account"
  8. arcdao "go-common/app/interface/main/app-show/dao/archive"
  9. adtdao "go-common/app/interface/main/app-show/dao/audit"
  10. rcmmndao "go-common/app/interface/main/app-show/dao/recommend"
  11. rgdao "go-common/app/interface/main/app-show/dao/region"
  12. reldao "go-common/app/interface/main/app-show/dao/relation"
  13. "go-common/app/interface/main/app-show/model"
  14. "go-common/app/interface/main/app-show/model/region"
  15. account "go-common/app/service/main/account/model"
  16. "go-common/app/service/main/archive/api"
  17. "go-common/app/service/main/archive/model/archive"
  18. relation "go-common/app/service/main/relation/model"
  19. "go-common/library/log"
  20. "go-common/library/sync/errgroup"
  21. )
  22. const (
  23. _initRank = "rank_key_%s_%d"
  24. )
  25. var (
  26. // 番剧 动画,音乐,舞蹈,游戏,科技,娱乐,鬼畜,电影,时尚, 生活,连载番剧(二级分区),国漫,影视,纪录片,国创相关,数码
  27. _tids = []int{13, 1, 3, 129, 4, 36, 5, 119, 23, 155, 160, 11, 33, 167, 181, 177, 168, 188}
  28. // region.ShowItem
  29. _emptyShowItems = []*region.ShowItem{}
  30. // _pgctids = map[int]struct{}{
  31. // 177: struct{}{},
  32. // }
  33. )
  34. type Service struct {
  35. c *conf.Config
  36. // region
  37. rdao *rgdao.Dao
  38. // rcmmnd
  39. rcmmnd *rcmmndao.Dao
  40. // archive
  41. arc *arcdao.Dao
  42. // audit
  43. adt *adtdao.Dao
  44. // account
  45. accd *accdao.Dao
  46. // relation
  47. reldao *reldao.Dao
  48. // tick
  49. tick time.Duration
  50. // ranking
  51. rankCache map[string][]*region.ShowItem
  52. rankOseaCache map[string][]*region.ShowItem
  53. // audit cache
  54. auditCache map[string]map[int]struct{} // audit mobi_app builds
  55. }
  56. // New new a region service.
  57. func New(c *conf.Config) (s *Service) {
  58. s = &Service{
  59. c: c,
  60. rdao: rgdao.New(c),
  61. rcmmnd: rcmmndao.New(c),
  62. // archive
  63. arc: arcdao.New(c),
  64. // audit
  65. adt: adtdao.New(c),
  66. // account
  67. accd: accdao.New(c),
  68. // relation
  69. reldao: reldao.New(c),
  70. // tick
  71. tick: time.Duration(c.Tick),
  72. // ranking
  73. rankCache: map[string][]*region.ShowItem{},
  74. rankOseaCache: map[string][]*region.ShowItem{},
  75. // audit cache
  76. auditCache: map[string]map[int]struct{}{},
  77. }
  78. time.Sleep(time.Second * 2)
  79. s.load()
  80. s.loadAuditCache()
  81. go s.loadproc()
  82. return
  83. }
  84. // RankShow
  85. func (s *Service) RankShow(c context.Context, plat int8, rid, pn, ps int, mid int64, order string) (res []*region.ShowItem) {
  86. var (
  87. key = fmt.Sprintf(_initRank, order, rid)
  88. tmp []*region.ShowItem
  89. authorMids []int64
  90. authorMidExist = map[int64]struct{}{}
  91. authorRelations map[int64]*account.Relation
  92. authorStats map[int64]*relation.Stat
  93. authorCards map[int64]*account.Card
  94. err error
  95. )
  96. if model.IsOverseas(plat) {
  97. tmp = s.rankOseaCache[key]
  98. } else {
  99. tmp = s.rankCache[key]
  100. }
  101. start := (pn - 1) * ps
  102. end := start + ps
  103. if len(tmp) == 0 || start > len(tmp) {
  104. res = _emptyShowItems
  105. return
  106. }
  107. if end < len(tmp) {
  108. tmp = tmp[start:end]
  109. } else {
  110. tmp = tmp[start:]
  111. }
  112. for _, t := range tmp {
  113. i := &region.ShowItem{}
  114. *i = *t
  115. res = append(res, i)
  116. // up mid
  117. if _, ok := authorMidExist[i.Mid]; !ok && i.Mid > 0 {
  118. authorMids = append(authorMids, i.Mid)
  119. authorMidExist[i.Mid] = struct{}{}
  120. }
  121. }
  122. if len(authorMids) > 0 {
  123. g, ctx := errgroup.WithContext(c)
  124. g.Go(func() error {
  125. if authorCards, err = s.accd.Cards3(ctx, authorMids); err != nil {
  126. log.Error("s.accd.Cards3 error(%v)", err)
  127. }
  128. return nil
  129. })
  130. if mid > 0 {
  131. g.Go(func() error {
  132. if authorRelations, err = s.accd.Relations3(ctx, mid, authorMids); err != nil {
  133. log.Error("s.accd.Relations2 error(%v)", err)
  134. }
  135. return nil
  136. })
  137. }
  138. g.Go(func() error {
  139. if authorStats, err = s.reldao.Stats(ctx, authorMids); err != nil {
  140. log.Error("s.reldao.Stats error(%v)", err)
  141. }
  142. return nil
  143. })
  144. if err = g.Wait(); err != nil {
  145. log.Error("RankUser errgroup.WithContext error(%v)", err)
  146. }
  147. }
  148. for _, i := range res {
  149. if len(authorRelations) > 0 {
  150. if relations, ok := authorRelations[i.Mid]; ok {
  151. if relations.Following {
  152. i.Attribute = 1
  153. }
  154. }
  155. }
  156. if len(authorStats) > 0 {
  157. if stats, ok := authorStats[i.Mid]; ok {
  158. i.Follower = int(stats.Follower)
  159. }
  160. }
  161. if len(authorCards) > 0 {
  162. if info, ok := authorCards[i.Mid]; ok {
  163. ov := &region.OfficialVerify{}
  164. ov.FromOfficialVerify(info.Official)
  165. i.OfficialVerify = ov
  166. }
  167. }
  168. if !model.IsIPad(plat) {
  169. if i.RedirectURL != "" {
  170. i.URI = i.RedirectURL
  171. i.Goto = model.GotoBangumi
  172. }
  173. }
  174. }
  175. return
  176. }
  177. // loadproc
  178. func (s *Service) loadproc() {
  179. for {
  180. time.Sleep(s.tick)
  181. s.load()
  182. s.loadAuditCache()
  183. }
  184. }
  185. // load load Rank all
  186. func (s *Service) load() {
  187. var (
  188. tmp = map[string][]*region.ShowItem{}
  189. tmpOsea = map[string][]*region.ShowItem{}
  190. )
  191. for _, rid := range _tids {
  192. if rid == 33 {
  193. aids, others, scores, err := s.rcmmnd.RankAppBangumi(context.TODO())
  194. key := fmt.Sprintf(_initRank, "bangumi", 0)
  195. if err != nil || len(aids) < 5 {
  196. log.Error("s.rcmmnd.RankAppBangumi len lt 20 OR error(%v)", err)
  197. tmp[key], tmpOsea[key] = s.rankCache[key], s.rankOseaCache[key]
  198. continue
  199. }
  200. tmp[key], tmpOsea[key] = s.fromRankAids(context.TODO(), aids, others, scores)
  201. log.Info("loadRankBangumi success")
  202. } else {
  203. aids, others, scores, err := s.rcmmnd.RankAppRegion(context.TODO(), rid)
  204. key := fmt.Sprintf(_initRank, "all", rid)
  205. if err != nil || len(aids) < 5 {
  206. log.Error("s.rcmmnd.RankAppRegion rid (%v) len lt 20 OR error(%v)", rid, err)
  207. tmp[key], tmpOsea[key] = s.rankCache[key], s.rankOseaCache[key]
  208. continue
  209. }
  210. tmp[key], tmpOsea[key] = s.fromRankAids(context.TODO(), aids, others, scores)
  211. log.Info("loadRankRegion(%s_%d) success", "all", rid)
  212. }
  213. }
  214. aids, others, scores, err := s.rcmmnd.RankAppAll(context.TODO())
  215. key := fmt.Sprintf(_initRank, "all", 0)
  216. if err != nil || len(aids) < 5 {
  217. log.Error("s.rcmmnd.RankAppAll(%s) len lt 20 OR error(%v)", "all", err)
  218. return
  219. }
  220. tmp[key], tmpOsea[key] = s.fromRankAids(context.TODO(), aids, others, scores)
  221. log.Info("loadRank(%s) success", "all")
  222. aids, others, scores, err = s.rcmmnd.RankAppOrigin(context.TODO())
  223. key = fmt.Sprintf(_initRank, "origin", 0)
  224. if err != nil || len(aids) < 5 {
  225. log.Error("s.rcmmnd.RankAppOrigin(%s) len lt 20 OR error(%v)", "all", err)
  226. return
  227. }
  228. tmp[key], tmpOsea[key] = s.fromRankAids(context.TODO(), aids, others, scores)
  229. log.Info("loadRank(%s) success", "origin")
  230. if len(tmp) > 0 {
  231. s.rankCache = tmp
  232. }
  233. if len(tmpOsea) > 0 {
  234. s.rankOseaCache = tmpOsea
  235. }
  236. }
  237. // fromRankAids
  238. func (s *Service) fromRankAids(ctx context.Context, aids []int64, others, scores map[int64]int64) (sis, sisOsea []*region.ShowItem) {
  239. var (
  240. aid int64
  241. as map[int64]*api.Arc
  242. arc *api.Arc
  243. ok bool
  244. err error
  245. paid int64
  246. )
  247. if as, err = s.arc.ArchivesPB(ctx, aids); err != nil {
  248. log.Error("s.arc.ArchivesPB error(%v)", err)
  249. return
  250. }
  251. if len(as) == 0 {
  252. log.Warn("s.arc.ArchivesPB(%v) length is 0", aids)
  253. return
  254. }
  255. child := map[int64][]*region.ShowItem{}
  256. childOsea := map[int64][]*region.ShowItem{}
  257. for _, aid = range aids {
  258. if arc, ok = as[aid]; ok {
  259. if paid, ok = others[arc.Aid]; ok {
  260. i := &region.ShowItem{}
  261. i.FromArchivePBRank(arc, scores)
  262. child[paid] = append(child[paid], i)
  263. if arc.AttrVal(archive.AttrBitOverseaLock) == 0 {
  264. childOsea[paid] = append(childOsea[paid], i)
  265. }
  266. }
  267. }
  268. }
  269. for _, aid = range aids {
  270. if arc, ok = as[aid]; ok {
  271. if _, ok = others[arc.Aid]; !ok {
  272. i := &region.ShowItem{}
  273. i.FromArchivePBRank(arc, scores)
  274. if arc.AttrVal(archive.AttrBitOverseaLock) == 0 {
  275. if tmpchild, ok := childOsea[arc.Aid]; ok {
  276. i.Children = tmpchild
  277. }
  278. sisOsea = append(sisOsea, i)
  279. }
  280. if tmpchild, ok := child[arc.Aid]; ok {
  281. i.Children = tmpchild
  282. }
  283. sis = append(sis, i)
  284. }
  285. }
  286. }
  287. return
  288. }