show.go 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926
  1. package region
  2. import (
  3. "context"
  4. "fmt"
  5. "time"
  6. "go-common/app/interface/main/app-show/model"
  7. seasongrpc "go-common/app/service/openplatform/pgc-season/api/grpc/season/v1"
  8. "go-common/app/interface/main/app-show/model/region"
  9. "go-common/app/service/main/archive/api"
  10. "go-common/app/service/main/archive/model/archive"
  11. "go-common/library/log"
  12. "go-common/library/net/metadata"
  13. xtime "go-common/library/time"
  14. )
  15. const (
  16. _initCardKey = "card_key_%d_%d"
  17. _bangumiSeasonID = 1
  18. _bangumiEpisodeID = 2
  19. )
  20. var (
  21. _emptyShow = &region.Show{}
  22. _emptyShowItems = []*region.ShowItem{}
  23. _bangumiRids = map[int]struct{}{
  24. 33: struct{}{},
  25. 32: struct{}{},
  26. 153: struct{}{},
  27. 51: struct{}{},
  28. 152: struct{}{},
  29. 168: struct{}{},
  30. 169: struct{}{},
  31. 170: struct{}{},
  32. }
  33. _bangumiReids = map[int]struct{}{
  34. 13: struct{}{},
  35. 167: struct{}{},
  36. }
  37. )
  38. // Show region show
  39. func (s *Service) Show(c context.Context, plat int8, rid, build int, mid int64, channel, buvid, network, mobiApp, device, adExtra string) (res *region.Show) {
  40. ip := metadata.String(c, metadata.RemoteIP)
  41. if model.IsOverseas(plat) {
  42. res = s.showOseaCache[rid]
  43. } else {
  44. res = s.showCache[rid]
  45. }
  46. if res == nil {
  47. res = _emptyShow
  48. return
  49. }
  50. if !model.IsIPad(plat) && len(res.Recommend) >= 4 {
  51. res = &region.Show{
  52. Recommend: res.Recommend[:4],
  53. New: res.New,
  54. Dynamic: res.Dynamic,
  55. }
  56. } else if model.IsIPad(plat) && len(res.Recommend) < 8 {
  57. var (
  58. max = 8
  59. hotlen = len(res.Recommend)
  60. )
  61. if last := max - hotlen; last < len(res.New) {
  62. res.Recommend = append(res.Recommend, res.New[:last]...)
  63. } else if len(res.New) > 0 {
  64. res.Recommend = append(res.Recommend, res.New...)
  65. } else {
  66. if last < len(res.Dynamic) {
  67. res.Recommend = append(res.Recommend, res.Dynamic[:last]...)
  68. } else if len(res.Dynamic) > 0 {
  69. res.Recommend = append(res.Recommend, res.Dynamic...)
  70. }
  71. }
  72. }
  73. res.Banner = s.getBanners(c, plat, build, rid, mid, channel, ip, buvid, network, mobiApp, device, adExtra)
  74. return
  75. }
  76. // ShowDynamic show dynamic page
  77. func (s *Service) ShowDynamic(c context.Context, plat int8, build, rid, pn, ps int) (res []*region.ShowItem) {
  78. var (
  79. isOsea = model.IsOverseas(plat) //is overseas
  80. bangumiType = 0
  81. )
  82. if _, isBangumi := _bangumiReids[rid]; isBangumi {
  83. if (plat == model.PlatIPhone && build > 6090) || (plat == model.PlatAndroid && build > 514000) {
  84. bangumiType = _bangumiEpisodeID
  85. }
  86. }
  87. start := (pn - 1) * ps
  88. end := start + ps
  89. if bangumiType != 0 {
  90. if end < len(s.showDynamicAidsCache[rid]) {
  91. aids := s.showDynamicAidsCache[rid][start:end]
  92. res = s.fromAidsOsea(c, aids, false, isOsea, bangumiType)
  93. if len(res) > 0 {
  94. return
  95. }
  96. }
  97. }
  98. if !isOsea {
  99. if end < len(s.dynamicCache[rid]) {
  100. res = s.dynamicCache[rid][start:end]
  101. return
  102. }
  103. } else {
  104. if end < len(s.dynamicOseaCache[rid]) {
  105. res = s.dynamicOseaCache[rid][start:end]
  106. return
  107. }
  108. }
  109. as, aids, err := s.dyn.RegionDynamic(c, rid, pn, ps)
  110. if err != nil {
  111. log.Error("s.rcmmnd.RegionDynamic(%d, %d, %d) error(%v)", rid, pn, ps, err)
  112. return
  113. }
  114. if bangumiType != 0 {
  115. res = s.fromArchivesPBBangumiOsea(c, as, aids, isOsea, bangumiType)
  116. } else {
  117. res = s.fromArchivesPBOsea(as, isOsea)
  118. }
  119. return
  120. }
  121. // ChildShow region child show
  122. func (s *Service) ChildShow(c context.Context, plat int8, mid int64, rid, tid, build int, channel, mobiApp string, now time.Time) (res *region.Show) {
  123. var (
  124. isOsea = model.IsOverseas(plat) //is overseas
  125. bangumiType = 0
  126. max = 20
  127. )
  128. if _, isBangumi := _bangumiRids[rid]; isBangumi {
  129. if (plat == model.PlatIPhone && build > 6050) || (plat == model.PlatAndroid && build > 512007) {
  130. bangumiType = _bangumiEpisodeID
  131. } else {
  132. bangumiType = _bangumiSeasonID
  133. }
  134. }
  135. if (mobiApp == "iphone" && build <= 4280) || (mobiApp == "ipad" && build <= 10400) || (mobiApp == "white" && build <= 101320) ||
  136. (mobiApp == "android" && build <= 501020) || (mobiApp == "android_tv" && build <= 1310) || mobiApp == "android_G" || mobiApp == "win" {
  137. bangumiType = 0
  138. }
  139. if tid == 0 {
  140. if bangumiType != 0 {
  141. var (
  142. hotTmp, newTmp []*region.ShowItem
  143. aids []int64
  144. hotOk, newOk bool
  145. )
  146. if aids, hotOk = s.childHotAidsCache[rid]; hotOk {
  147. if len(aids) > max {
  148. aids = aids[:max]
  149. }
  150. hotTmp = s.fromAidsOsea(c, aids, false, isOsea, bangumiType)
  151. }
  152. if aids, newOk = s.childNewAidsCache[rid]; newOk {
  153. if len(aids) > max {
  154. aids = aids[:max]
  155. }
  156. newTmp = s.fromAidsOsea(c, aids, false, isOsea, bangumiType)
  157. }
  158. if hotOk && newOk && len(hotTmp) > 0 && len(newTmp) > 0 {
  159. res, _ = s.mergeChildShow(hotTmp, newTmp)
  160. return
  161. }
  162. }
  163. if !isOsea {
  164. if res = s.childShowCache[rid]; res == nil {
  165. res = _emptyShow
  166. return
  167. }
  168. } else {
  169. if res = s.childShowOseaCache[rid]; res == nil {
  170. res = _emptyShow
  171. return
  172. }
  173. }
  174. return
  175. }
  176. res = s.mergeTagShow(c, rid, tid, bangumiType, isOsea, now)
  177. if mid > 0 {
  178. var err error
  179. if res.Tag, err = s.tag.TagInfo(c, mid, tid, now); err != nil {
  180. log.Error("s.tag.TagInfo(%d, %d) error(%v)", mid, tid, err)
  181. }
  182. }
  183. return
  184. }
  185. // ChildListShow region childList show
  186. func (s *Service) ChildListShow(c context.Context, plat int8, rid, tid, pn, ps, build int, mid int64, order, platform, mobiApp, device string) (res []*region.ShowItem) {
  187. ip := metadata.String(c, metadata.RemoteIP)
  188. var (
  189. isOsea = model.IsOverseas(plat) //is overseas
  190. bangumiType = 0
  191. )
  192. start := (pn - 1) * ps
  193. end := start + ps
  194. key := fmt.Sprintf(_initRegionTagKey, rid, tid)
  195. if _, isBangumi := _bangumiRids[rid]; isBangumi {
  196. if (plat == model.PlatIPhone && build > 6050) || (plat == model.PlatAndroid && build > 512007) {
  197. bangumiType = _bangumiEpisodeID
  198. } else {
  199. bangumiType = _bangumiSeasonID
  200. }
  201. }
  202. if (mobiApp == "iphone" && build <= 4280) || (mobiApp == "ipad" && build <= 10400) || (mobiApp == "white" && build <= 101320) ||
  203. (mobiApp == "android" && build <= 501020) || (mobiApp == "android_tv" && build <= 1310) || mobiApp == "android_G" || mobiApp == "win" {
  204. bangumiType = 0
  205. }
  206. if bangumiType != 0 {
  207. if tid == 0 && (order == "" || order == "new") && end < len(s.childNewAidsCache[rid]) {
  208. aids := s.childNewAidsCache[rid][start:end]
  209. res = s.fromAidsOsea(c, aids, false, isOsea, bangumiType)
  210. if len(res) > 0 {
  211. return
  212. }
  213. }
  214. if tid > 0 && (order == "" || order == "new") && end < len(s.tagNewAidsCache[key]) {
  215. aids := s.tagNewAidsCache[key][start:end]
  216. res = s.fromAidsOsea(c, aids, false, isOsea, bangumiType)
  217. if len(res) > 0 {
  218. return
  219. }
  220. }
  221. }
  222. if !isOsea {
  223. if tid == 0 && (order == "" || order == "new") && end < len(s.childNewCache[rid]) {
  224. res = s.childNewCache[rid][start:end]
  225. return
  226. }
  227. if tid > 0 && (order == "" || order == "new") && end < len(s.tagNewCache[key]) {
  228. res = s.tagNewCache[key][start:end]
  229. return
  230. }
  231. } else {
  232. if tid == 0 && (order == "" || order == "new") && end < len(s.childNewOseaCache[rid]) {
  233. res = s.childNewOseaCache[rid][start:end]
  234. return
  235. }
  236. if tid > 0 && (order == "" || order == "new") && end < len(s.tagNewOseaCache[key]) {
  237. res = s.tagNewOseaCache[key][start:end]
  238. return
  239. }
  240. }
  241. if (order == "" || order == "new") && tid == 0 {
  242. arcs, aids, err := s.arc.RanksArcs(c, rid, pn, ps)
  243. if err != nil {
  244. log.Error("s.rcmmnd.RegionArcList(%d, %d, %d, %d) error(%v)", rid, pn, ps, err)
  245. }
  246. if bangumiType != 0 {
  247. res = s.fromArchivesPBBangumiOsea(c, arcs, aids, isOsea, bangumiType)
  248. } else {
  249. res = s.fromArchivesPBOsea(arcs, isOsea)
  250. }
  251. return
  252. } else if (order == "" || order == "new") && tid > 0 {
  253. as, err := s.tag.NewArcs(c, rid, tid, pn, ps, time.Now())
  254. if err != nil {
  255. log.Error("s.tag.NewArcs(%d, %d) error(%v)", rid, tid, err)
  256. return
  257. }
  258. res = s.fromAidsOsea(c, as, false, isOsea, bangumiType)
  259. return
  260. }
  261. var (
  262. tname string
  263. ok bool
  264. )
  265. if tid > 0 {
  266. if tname, ok = s.tagsCache[key]; !ok {
  267. return
  268. }
  269. }
  270. as, err := s.search.SearchList(c, rid, build, pn, ps, mid, time.Now(), ip, order, tname, platform, mobiApp, device)
  271. if err != nil {
  272. log.Error("s.search.SearchList(%d, %d, %v, %d, %d) error(%v)", rid, tid, tname, pn, ps, err)
  273. }
  274. res = s.fromAidsOsea(c, as, false, isOsea, bangumiType)
  275. return
  276. }
  277. // Dynamic region dynamic
  278. func (s *Service) Dynamic(c context.Context, plat int8, rid, build int, mid int64, channel, buvid, network, mobiApp, device, adExtra string, now time.Time) (res *region.Show) {
  279. ip := metadata.String(c, metadata.RemoteIP)
  280. var (
  281. isOsea = model.IsOverseas(plat) //is overseas
  282. resCache *region.Show
  283. )
  284. s.prmobi.Incr("region_dynamic_plat_" + mobiApp)
  285. if isOsea {
  286. if resCache = s.regionFeedOseaCache[rid]; resCache == nil {
  287. resCache = _emptyShow
  288. }
  289. } else {
  290. if resCache = s.regionFeedCache[rid]; resCache == nil {
  291. resCache = _emptyShow
  292. }
  293. }
  294. if dyn, err := s.feedRegionDynamic(c, plat, rid, 0, 0, false, true, 0, mid, resCache.Recommend, now); err == nil && dyn != nil {
  295. res = dyn
  296. } else {
  297. res = resCache
  298. }
  299. if res != nil {
  300. res.Banner = s.getBanners(c, plat, build, rid, mid, channel, ip, buvid, network, mobiApp, device, adExtra)
  301. res.Card = s.regionCardDisplay(plat, build, rid)
  302. }
  303. return
  304. }
  305. // regionCardDisplay
  306. func (s *Service) regionCardDisplay(plat int8, build, rid int) (res []*region.Head) {
  307. var ss []*region.Head
  308. key := fmt.Sprintf(_initCardKey, plat, rid)
  309. ss = s.cardCache[key]
  310. if len(ss) == 0 {
  311. res = []*region.Head{}
  312. return
  313. }
  314. res = []*region.Head{}
  315. for _, sw := range ss {
  316. if model.InvalidBuild(build, sw.Build, sw.Condition) {
  317. continue
  318. }
  319. tmp := &region.Head{}
  320. *tmp = *sw
  321. tmp.FillBuildURI(plat, build)
  322. res = append(res, tmp)
  323. }
  324. return
  325. }
  326. // DynamicList show dynamic list
  327. func (s *Service) DynamicList(c context.Context, plat int8, rid int, pull bool, ctime, mid int64, now time.Time) (res *region.Show) {
  328. var (
  329. err error
  330. )
  331. if res, err = s.feedRegionDynamic(c, plat, rid, 0, 0, pull, false, ctime, mid, nil, now); err != nil || res == nil {
  332. res = _emptyShow
  333. return
  334. }
  335. return
  336. }
  337. // DynamicChild region show dynamic
  338. func (s *Service) DynamicChild(c context.Context, plat int8, rid, tid, build int, mid int64, mobiApp string, now time.Time) (res *region.Show) {
  339. var (
  340. isOsea = model.IsOverseas(plat) //is overseas
  341. bangumiType = 0
  342. resCache *region.Show
  343. max = 20
  344. )
  345. s.prmobi.Incr("region_dynamic_child_plat_" + mobiApp)
  346. if _, isBangumi := _bangumiRids[rid]; isBangumi {
  347. if (plat == model.PlatIPhone && build > 6050) || (plat == model.PlatAndroid && build > 512007) {
  348. bangumiType = _bangumiEpisodeID
  349. } else {
  350. bangumiType = _bangumiSeasonID
  351. }
  352. }
  353. if tid == 0 {
  354. if bangumiType != 0 {
  355. var (
  356. hotTmp, newTmp []*region.ShowItem
  357. aids []int64
  358. hotOk, newOk bool
  359. )
  360. if aids, hotOk = s.childHotAidsCache[rid]; hotOk {
  361. if len(aids) > max {
  362. aids = aids[:max]
  363. }
  364. hotTmp = s.fromAidsOsea(c, aids, false, isOsea, bangumiType)
  365. }
  366. if aids, newOk = s.childNewAidsCache[rid]; newOk {
  367. if len(aids) > max {
  368. aids = aids[:max]
  369. }
  370. newTmp = s.fromAidsOsea(c, aids, false, isOsea, bangumiType)
  371. }
  372. if len(hotTmp) > 0 && len(newTmp) > 0 {
  373. resCache, _ = s.mergeChildShow(hotTmp, newTmp)
  374. }
  375. }
  376. if resCache == nil {
  377. if !isOsea {
  378. if resCache = s.childShowCache[rid]; resCache == nil {
  379. resCache = _emptyShow
  380. }
  381. } else {
  382. if resCache = s.childShowOseaCache[rid]; resCache == nil {
  383. resCache = _emptyShow
  384. }
  385. }
  386. }
  387. } else {
  388. resCache = s.mergeTagShow(c, rid, tid, bangumiType, isOsea, now)
  389. }
  390. if resCache == nil {
  391. resCache = _emptyShow
  392. }
  393. if dyn, err := s.feedRegionDynamic(c, plat, rid, tid, bangumiType, false, true, 0, mid, nil, now); err == nil && dyn != nil {
  394. res = dyn
  395. s.pHit.Incr("feed_region_dynamic")
  396. } else {
  397. res = resCache
  398. s.pMiss.Incr("feed_region_dynamic")
  399. }
  400. if res != nil {
  401. if tid == 0 {
  402. if tags, ok := s.regionTagCache[rid]; ok {
  403. res.TopTag = tags
  404. }
  405. } else if mid > 0 {
  406. var err error
  407. if res.Tag, err = s.tag.TagInfo(c, mid, tid, now); err != nil {
  408. log.Error("s.tag.TagInfo(%d, %d) error(%v)", mid, tid, err)
  409. }
  410. }
  411. }
  412. return
  413. }
  414. // DynamicListChild dynamic childList show
  415. func (s *Service) DynamicListChild(c context.Context, plat int8, rid, tid, build int, pull bool, ctime, mid int64, now time.Time) (res *region.Show) {
  416. var (
  417. err error
  418. bangumiType = 0
  419. )
  420. if _, isBangumi := _bangumiRids[rid]; isBangumi {
  421. if (plat == model.PlatIPhone && build > 6050) || (plat == model.PlatAndroid && build > 512007) {
  422. bangumiType = _bangumiEpisodeID
  423. } else {
  424. bangumiType = _bangumiSeasonID
  425. }
  426. }
  427. if res, err = s.feedRegionDynamic(c, plat, rid, tid, bangumiType, pull, false, ctime, mid, nil, now); err != nil || res == nil {
  428. res = _emptyShow
  429. return
  430. }
  431. return
  432. }
  433. // feedRegionDynamic
  434. func (s *Service) feedRegionDynamic(c context.Context, plat int8, rid, tid, bangumiType int, pull, isRecommend bool, ctime, mid int64, items []*region.ShowItem, now time.Time) (res *region.Show, err error) {
  435. var (
  436. smTagNum = 10
  437. smTagPos = 4
  438. )
  439. if res, err = s.feedDynamic(c, plat, rid, tid, bangumiType, pull, isRecommend, ctime, mid, items, now); err != nil || res == nil {
  440. log.Error("s.feedDynamic is null rid:%v tid:%v", rid, tid)
  441. } else {
  442. if isRecommend && tid > 0 {
  443. if st := s.similarTags(c, rid, tid, now); st != nil {
  444. if len(st) > smTagNum {
  445. res.TopTag = st[:smTagNum]
  446. res.NewTag = &region.NewTag{
  447. Position: smTagPos,
  448. Tag: st[smTagNum:],
  449. }
  450. } else {
  451. res.TopTag = st
  452. }
  453. }
  454. }
  455. }
  456. return
  457. }
  458. // feedRegionDynamic
  459. func (s *Service) feedDynamic(c context.Context, plat int8, rid, tid, bangumiType int, pull, isRecommend bool, ctime, mid int64, items []*region.ShowItem, now time.Time) (res *region.Show, err error) {
  460. var (
  461. isOsea = model.IsOverseas(plat) //is overseas
  462. newAids = []int64{}
  463. hotAids = []int64{}
  464. ctop, cbottom xtime.Time
  465. )
  466. if hotAids, newAids, ctop, cbottom, err = s.rcmmnd.FeedDynamic(c, pull, rid, tid, ctime, mid, now); err != nil || len(newAids) == 0 {
  467. log.Error("s.rcmmnd.FeedDynamic(%v) error(%v)", rid, err)
  468. return
  469. }
  470. newItems := s.fromAidsOsea(c, newAids, false, isOsea, bangumiType)
  471. hotItems := s.fromAidsOsea(c, hotAids, false, isOsea, bangumiType)
  472. if isRecommend && (len(newItems) > 0 || len(hotItems) > 4) && len(newItems) > 0 {
  473. if len(hotItems) >= 4 {
  474. res = s.mergeDynamicShow(hotItems, newItems, ctop, cbottom)
  475. } else {
  476. log.Error("feedDynamic_hot is null rid:%v tid:%v", rid, tid)
  477. res = s.mergeDynamicShow(items, newItems, ctop, cbottom)
  478. }
  479. } else if !isRecommend && len(newItems) > 0 {
  480. res = s.mergeDynamicShow(_emptyShowItems, newItems, ctop, cbottom)
  481. } else {
  482. log.Error("feedDynamic_newItems is null rid:%v tid:%v", rid, tid)
  483. }
  484. s.infoc(mid, hotAids, newAids, rid, tid, pull, now)
  485. return
  486. }
  487. // fromArchives return region show items from archive archives.
  488. func (s *Service) fromArchivesPB(as []*api.Arc) (is, isOsea []*region.ShowItem) {
  489. var asLen = len(as)
  490. if asLen == 0 {
  491. is = _emptyShowItems
  492. return
  493. }
  494. is = make([]*region.ShowItem, 0, asLen)
  495. for _, a := range as {
  496. i := &region.ShowItem{}
  497. i.FromArchivePB(a)
  498. if a.AttrVal(archive.AttrBitOverseaLock) == 0 {
  499. isOsea = append(isOsea, i)
  500. }
  501. is = append(is, i)
  502. }
  503. return
  504. }
  505. // fromArchivesPBBangumi aid to sid
  506. func (s *Service) fromArchivesPBBangumi(c context.Context, as []*api.Arc, aids []int64, bangumiType int) (is, isOsea []*region.ShowItem) {
  507. var (
  508. asLen = len(as)
  509. err error
  510. // bangumi
  511. // sids map[int64]int64
  512. sids map[int32]*seasongrpc.CardInfoProto
  513. )
  514. if asLen == 0 {
  515. is = _emptyShowItems
  516. return
  517. }
  518. if sids, err = s.fromSeasonID(c, aids); err != nil {
  519. log.Error("s.fromSeasonID error(%v)", err)
  520. return
  521. }
  522. is = make([]*region.ShowItem, 0, asLen)
  523. for _, a := range as {
  524. if sid, ok := sids[int32(a.Aid)]; ok && sid.SeasonId != 0 {
  525. i := &region.ShowItem{}
  526. i.FromBangumiArchivePB(a, sid, bangumiType)
  527. if a.AttrVal(archive.AttrBitOverseaLock) == 0 {
  528. isOsea = append(isOsea, i)
  529. }
  530. is = append(is, i)
  531. }
  532. }
  533. return
  534. }
  535. // mergeShow merge show
  536. func (s *Service) mergeShow(hotTmp, newTmp, dynTmp []*region.ShowItem) (rs *region.Show) {
  537. rs = &region.Show{}
  538. if len(hotTmp) >= 4 {
  539. rs.Recommend = hotTmp
  540. } else {
  541. rs.Recommend = _emptyShowItems
  542. }
  543. if len(newTmp) >= 4 {
  544. rs.New = newTmp[:4]
  545. } else {
  546. rs.New = _emptyShowItems
  547. }
  548. if len(dynTmp) > 20 {
  549. rs.Dynamic = dynTmp[:20]
  550. } else if len(dynTmp) > 0 {
  551. rs.Dynamic = dynTmp
  552. } else {
  553. rs.Dynamic = _emptyShowItems
  554. }
  555. return
  556. }
  557. // mergeChildShow merge child show
  558. func (s *Service) mergeChildShow(recTmp, newTmp []*region.ShowItem) (rs *region.Show, new []*region.ShowItem) {
  559. var (
  560. last int
  561. )
  562. rs = &region.Show{}
  563. if len(recTmp) >= 4 {
  564. rs.Recommend = recTmp[:4]
  565. } else if len(newTmp) >= 4 {
  566. var (
  567. max = 4
  568. hotlen = len(recTmp)
  569. )
  570. rs.Recommend = recTmp
  571. if last = max - hotlen; last < len(newTmp) {
  572. rs.Recommend = append(rs.Recommend, newTmp[:last]...)
  573. } else {
  574. rs.Recommend = append(rs.Recommend, newTmp...)
  575. }
  576. } else {
  577. rs.Recommend = _emptyShowItems
  578. }
  579. new = newTmp
  580. if last > 0 {
  581. if len(new) > last {
  582. new = new[last:]
  583. }
  584. }
  585. if len(new) >= 20 {
  586. rs.New = new[:20]
  587. } else if len(new) > 0 {
  588. rs.New = new
  589. } else {
  590. rs.New = _emptyShowItems
  591. }
  592. return
  593. }
  594. // mergeDynamicShow
  595. func (s *Service) mergeDynamicShow(recTmp, dynTmp []*region.ShowItem, ctop, cbottom xtime.Time) (rs *region.Show) {
  596. if ctop == 0 || cbottom == 0 {
  597. rs = &region.Show{}
  598. } else {
  599. rs = &region.Show{
  600. Ctop: ctop,
  601. Cbottom: cbottom,
  602. }
  603. }
  604. if len(recTmp) >= 4 {
  605. rs.Recommend = recTmp[:4]
  606. } else {
  607. rs.Recommend = _emptyShowItems
  608. }
  609. if len(dynTmp) > 0 {
  610. rs.New = dynTmp
  611. } else {
  612. rs.New = _emptyShowItems
  613. }
  614. return
  615. }
  616. // mergeTagShow merge tag show
  617. func (s *Service) mergeTagShow(c context.Context, rid, tid, bangumiType int, isOsea bool, now time.Time) (rs *region.Show) {
  618. const (
  619. strtNum = 1
  620. hotNum = 4
  621. newNum = 20
  622. maxNum = 50
  623. smTagNum = 10
  624. smTagPos = 4
  625. )
  626. rs = &region.Show{}
  627. // hots
  628. key := fmt.Sprintf(_initRegionTagKey, rid, tid)
  629. var (
  630. is []*region.ShowItem
  631. ok = false
  632. )
  633. if bangumiType != 0 {
  634. var aids []int64
  635. if aids, ok = s.tagHotAidsCache[key]; ok {
  636. is = s.fromAidsOsea(c, aids, false, isOsea, bangumiType)
  637. }
  638. } else if bangumiType == 0 || !ok {
  639. is, ok = s.tagHotCache[key]
  640. if isOsea {
  641. is, ok = s.tagHotOseaCache[key]
  642. }
  643. }
  644. if !ok {
  645. arcAids, err := s.tag.Hots(c, rid, tid, strtNum, hotNum, now)
  646. if err != nil {
  647. log.Error("s.tag.Hots(%d, %d) error(%v)", rid, tid, err)
  648. } else if is = s.fromAidsOsea(c, arcAids, true, isOsea, bangumiType); len(is) > 0 {
  649. if len(is) > hotNum {
  650. is = is[:hotNum]
  651. }
  652. }
  653. }
  654. rs.Recommend = is
  655. // news
  656. if bangumiType != 0 {
  657. var aids []int64
  658. if aids, ok = s.tagNewAidsCache[key]; ok {
  659. is = s.fromAidsOsea(c, aids, false, isOsea, bangumiType)
  660. }
  661. } else if bangumiType == 0 || !ok {
  662. is, ok = s.tagNewCache[key]
  663. if isOsea {
  664. is, ok = s.tagNewOseaCache[key]
  665. }
  666. }
  667. if !ok {
  668. as, err := s.tag.NewArcs(c, rid, tid, strtNum, maxNum, now)
  669. if err != nil {
  670. log.Error("s.tag.NewArcs(%d, %d) error(%v)", rid, tid, err)
  671. return
  672. }
  673. is = s.fromAidsOsea(c, as, false, isOsea, bangumiType)
  674. }
  675. var (
  676. hotlen = len(rs.Recommend)
  677. last int
  678. )
  679. if hotlen < hotNum {
  680. if last = hotNum - hotlen; last < len(is) {
  681. rs.Recommend = append(rs.Recommend, is[:last]...)
  682. } else {
  683. rs.Recommend = append(rs.Recommend, is...)
  684. }
  685. }
  686. if last > 0 {
  687. if len(is) > last {
  688. is = is[last:]
  689. }
  690. }
  691. if len(is) >= newNum {
  692. is = is[:newNum]
  693. }
  694. rs.New = is
  695. // similar tags
  696. if st := s.similarTags(c, rid, tid, now); st != nil {
  697. if len(st) > smTagNum {
  698. rs.TopTag = st[:smTagNum]
  699. rs.NewTag = &region.NewTag{
  700. Position: smTagPos,
  701. Tag: st[smTagNum+1:],
  702. }
  703. } else {
  704. rs.TopTag = st
  705. }
  706. }
  707. return
  708. }
  709. // fromAids get Aids. bangumiType 1 seasonid 2 epid
  710. func (s *Service) fromAids(c context.Context, arcAids []int64, isCheck bool, bangumiType int) (data, dataOsea []*region.ShowItem) {
  711. if len(arcAids) == 0 {
  712. return
  713. }
  714. var (
  715. ok bool
  716. aid int64
  717. arc *api.Arc
  718. as map[int64]*api.Arc
  719. err error
  720. // bangumi
  721. sids map[int32]*seasongrpc.CardInfoProto
  722. )
  723. if as, err = s.arc.ArchivesPB(c, arcAids); err != nil {
  724. log.Error("s.arc.ArchivesPB aids(%v) error(%v)", arcAids, err)
  725. return
  726. }
  727. data = make([]*region.ShowItem, 0, len(arcAids))
  728. if bangumiType != 0 {
  729. if sids, err = s.fromSeasonID(c, arcAids); err != nil {
  730. log.Error("s.fromSeasonID error(%v)", err)
  731. return
  732. }
  733. }
  734. for _, aid = range arcAids {
  735. if arc, ok = as[aid]; ok {
  736. if isCheck && !arc.IsNormal() {
  737. continue
  738. }
  739. i := &region.ShowItem{}
  740. if sid, ok := sids[int32(aid)]; ok && bangumiType != 0 && sid.SeasonId != 0 {
  741. i.FromBangumiArchivePB(arc, sid, bangumiType)
  742. } else {
  743. i.FromArchivePB(arc)
  744. }
  745. data = append(data, i)
  746. if arc.AttrVal(archive.AttrBitOverseaLock) == 0 {
  747. dataOsea = append(dataOsea, i)
  748. }
  749. }
  750. }
  751. return
  752. }
  753. // fromSeasonID
  754. func (s *Service) fromSeasonID(c context.Context, arcAids []int64) (seasonID map[int32]*seasongrpc.CardInfoProto, err error) {
  755. if seasonID, err = s.bgm.CardsByAids(c, arcAids); err != nil {
  756. log.Error("s.bmg.CardsByAids error %v", err)
  757. }
  758. return
  759. }
  760. // similarTags similar tags
  761. func (s *Service) similarTags(c context.Context, rid, tid int, now time.Time) (res []*region.SimilarTag) {
  762. key := fmt.Sprintf(_initRegionTagKey, rid, tid)
  763. res, ok := s.similarTagCache[key]
  764. if !ok {
  765. sts, err := s.tag.SimilarTag(c, rid, tid, now)
  766. if err != nil {
  767. log.Error("s.tag.SimilarTag error(%v)", err)
  768. return
  769. }
  770. res = sts
  771. }
  772. return
  773. }
  774. // isOverseas
  775. func (s *Service) fromAidsOsea(ctx context.Context, aids []int64, isCheck, isOsea bool, bangumiType int) (data []*region.ShowItem) {
  776. tmp, tmpOsea := s.fromAids(ctx, aids, isCheck, bangumiType)
  777. if isOsea {
  778. data = tmpOsea
  779. } else {
  780. data = tmp
  781. }
  782. if data == nil {
  783. data = _emptyShowItems
  784. }
  785. return
  786. }
  787. // fromArchivesOsea isOverseas
  788. func (s *Service) fromArchivesPBOsea(as []*api.Arc, isOsea bool) (data []*region.ShowItem) {
  789. tmp, tmpOsea := s.fromArchivesPB(as)
  790. if isOsea {
  791. data = tmpOsea
  792. } else {
  793. data = tmp
  794. }
  795. if data == nil {
  796. data = _emptyShowItems
  797. }
  798. return
  799. }
  800. // fromArchivesOsea isOverseas
  801. func (s *Service) fromArchivesPBBangumiOsea(c context.Context, as []*api.Arc, aids []int64, isOsea bool, bangumiType int) (data []*region.ShowItem) {
  802. tmp, tmpOsea := s.fromArchivesPBBangumi(c, as, aids, bangumiType)
  803. if isOsea {
  804. data = tmpOsea
  805. } else {
  806. data = tmp
  807. }
  808. if data == nil {
  809. data = _emptyShowItems
  810. }
  811. return
  812. }
  813. // fromRankAids
  814. func (s *Service) fromRankAids(ctx context.Context, aids []int64, others, scores map[int64]int64) (sis, sisOsea []*region.ShowItem) {
  815. var (
  816. aid int64
  817. as map[int64]*api.Arc
  818. arc *api.Arc
  819. ok bool
  820. err error
  821. paid int64
  822. )
  823. if as, err = s.arc.ArchivesPB(ctx, aids); err != nil {
  824. log.Error("s.arc.ArchivesPB aids(%v) error(%v)", aids, err)
  825. return
  826. }
  827. if len(as) == 0 {
  828. log.Warn("s.arc.ArchivesPB aids(%v) length is 0", aids)
  829. return
  830. }
  831. child := map[int64][]*region.ShowItem{}
  832. childOsea := map[int64][]*region.ShowItem{}
  833. for _, aid = range aids {
  834. if arc, ok = as[aid]; ok {
  835. if paid, ok = others[arc.Aid]; ok {
  836. i := &region.ShowItem{}
  837. i.FromArchivePBRank(arc, scores)
  838. child[paid] = append(child[paid], i)
  839. if arc.AttrVal(archive.AttrBitOverseaLock) == 0 {
  840. childOsea[paid] = append(childOsea[paid], i)
  841. }
  842. }
  843. }
  844. }
  845. for _, aid = range aids {
  846. if arc, ok = as[aid]; ok {
  847. if _, ok = others[arc.Aid]; !ok {
  848. i := &region.ShowItem{}
  849. i.FromArchivePBRank(arc, scores)
  850. if arc.AttrVal(archive.AttrBitOverseaLock) == 0 {
  851. if tmpchild, ok := childOsea[arc.Aid]; ok {
  852. i.Children = tmpchild
  853. }
  854. sisOsea = append(sisOsea, i)
  855. }
  856. if tmpchild, ok := child[arc.Aid]; ok {
  857. i.Children = tmpchild
  858. }
  859. sis = append(sis, i)
  860. }
  861. }
  862. }
  863. return
  864. }
  865. // fromCardAids get Aids.
  866. func (s *Service) fromCardAids(ctx context.Context, aids []int64) (data map[int64]*region.ShowItem) {
  867. var (
  868. arc *api.Arc
  869. ok bool
  870. )
  871. as, err := s.arc.ArchivesPB(ctx, aids)
  872. if err != nil {
  873. log.Error("s.arc.ArchivesPB error(%v)", err)
  874. return
  875. }
  876. if len(as) == 0 {
  877. log.Warn("s.arc.ArchivesPB(%v) length is 0", aids)
  878. return
  879. }
  880. data = map[int64]*region.ShowItem{}
  881. for _, aid := range aids {
  882. if arc, ok = as[aid]; ok {
  883. if !arc.IsNormal() {
  884. continue
  885. }
  886. i := &region.ShowItem{}
  887. i.FromArchivePB(arc)
  888. if region, ok := s.reRegionCache[int(arc.TypeID)]; ok {
  889. i.Reid = region.Rid
  890. }
  891. i.Desc = i.Rname
  892. data[aid] = i
  893. }
  894. }
  895. return
  896. }