daily.go 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406
  1. package daily
  2. import (
  3. "context"
  4. "fmt"
  5. "strconv"
  6. "time"
  7. "go-common/app/interface/main/app-show/conf"
  8. arcdao "go-common/app/interface/main/app-show/dao/archive"
  9. carddao "go-common/app/interface/main/app-show/dao/card"
  10. tagdao "go-common/app/interface/main/app-show/dao/tag"
  11. "go-common/app/interface/main/app-show/model"
  12. "go-common/app/interface/main/app-show/model/card"
  13. "go-common/app/interface/main/app-show/model/daily"
  14. "go-common/app/service/main/archive/api"
  15. "go-common/library/log"
  16. )
  17. const (
  18. _initDailyKey = "daily_key_%d_%d"
  19. _initColumnKey = "column_key_%d_%d"
  20. _initColumnListKey = "columnlist_key_%d_%d"
  21. )
  22. var (
  23. _emptyDaily = []*daily.Show{}
  24. _emptyList = []*daily.Item{}
  25. )
  26. type Service struct {
  27. c *conf.Config
  28. cdao *carddao.Dao
  29. arc *arcdao.Dao
  30. tag *tagdao.Dao
  31. // tick
  32. tick time.Duration
  33. // columnsCache
  34. columnsCache map[string]*card.Column
  35. // card
  36. cardCache map[string][]*daily.Show
  37. columnCache map[string]*daily.Show
  38. columnListCache map[string][]*daily.Item
  39. cardListCache map[string][]*card.ColumnList
  40. }
  41. // New new a daily service.
  42. func New(c *conf.Config) (s *Service) {
  43. s = &Service{
  44. c: c,
  45. cdao: carddao.New(c),
  46. arc: arcdao.New(c),
  47. tag: tagdao.New(c),
  48. // tick
  49. tick: time.Duration(c.Tick),
  50. // columnsCache
  51. columnsCache: map[string]*card.Column{},
  52. // card
  53. cardCache: map[string][]*daily.Show{},
  54. columnCache: map[string]*daily.Show{},
  55. columnListCache: map[string][]*daily.Item{},
  56. cardListCache: map[string][]*card.ColumnList{},
  57. }
  58. now := time.Now()
  59. s.loadColumnListCache(now)
  60. s.loadColumnsCache()
  61. s.loadNperCache(now)
  62. go s.cacheproc()
  63. return
  64. }
  65. // Daily
  66. func (s *Service) Daily(c context.Context, plat int8, build, dailyID, pn, ps int) (res []*daily.Show) {
  67. if pn > 0 {
  68. pn = pn - 1
  69. }
  70. start := pn * ps
  71. end := start + ps
  72. key := fmt.Sprintf(_initColumnKey, plat, dailyID)
  73. if column, ok := s.columnsCache[key]; ok {
  74. if model.InvalidBuild(build, column.Build, column.Condition) {
  75. res = _emptyDaily
  76. return
  77. }
  78. cardKey := fmt.Sprintf(_initDailyKey, plat, dailyID)
  79. if cards, ok := s.cardCache[cardKey]; ok {
  80. for _, sw := range cards {
  81. if model.InvalidBuild(build, sw.Build, sw.Condition) {
  82. continue
  83. }
  84. res = append(res, sw)
  85. }
  86. resLen := len(res)
  87. if resLen > end {
  88. res = res[start:end]
  89. } else if resLen > start {
  90. res = res[start:]
  91. } else {
  92. res = _emptyDaily
  93. }
  94. }
  95. }
  96. if len(res) == 0 {
  97. res = _emptyDaily
  98. }
  99. return
  100. }
  101. // ColumnList
  102. func (s *Service) ColumnList(plat int8, build, columnID int) (res *daily.ColumnList) {
  103. var (
  104. column []*daily.ColumnList
  105. )
  106. key := fmt.Sprintf(_initColumnListKey, plat, columnID)
  107. if columns, ok := s.cardListCache[key]; ok {
  108. for _, c := range columns {
  109. if model.InvalidBuild(build, c.Build, c.Condition) {
  110. continue
  111. }
  112. tmp := &daily.ColumnList{
  113. Cid: c.Cid,
  114. Name: c.Name,
  115. Ceid: c.Ceid,
  116. Cname: c.Cname,
  117. }
  118. column = append(column, tmp)
  119. }
  120. if len(column) > 0 {
  121. res = &daily.ColumnList{
  122. Ceid: column[0].Ceid,
  123. Name: column[0].Cname,
  124. Children: column,
  125. }
  126. }
  127. }
  128. return
  129. }
  130. // Category
  131. func (s *Service) Category(plat int8, build, categoryID, columnID, pn, ps int) (res *daily.Show) {
  132. var (
  133. key string
  134. )
  135. if pn > 0 {
  136. pn = pn - 1
  137. }
  138. start := pn * ps
  139. end := start + ps
  140. if columnID > 0 {
  141. key = fmt.Sprintf(_initDailyKey, plat, columnID)
  142. } else {
  143. listKey := fmt.Sprintf(_initColumnListKey, plat, categoryID)
  144. if columns, ok := s.cardListCache[listKey]; ok {
  145. for _, c := range columns {
  146. if model.InvalidBuild(build, c.Build, c.Condition) {
  147. continue
  148. }
  149. key = fmt.Sprintf(_initDailyKey, plat, c.Cid)
  150. break
  151. }
  152. }
  153. }
  154. if columns, ok := s.columnCache[key]; ok {
  155. res = columns
  156. if pn*ps > 400 {
  157. res.Body = _emptyList
  158. return
  159. }
  160. if res.Body, ok = s.columnListCache[key]; ok {
  161. resLen := len(res.Body)
  162. if resLen > end {
  163. res.Body = res.Body[start:end]
  164. } else if resLen > start {
  165. res.Body = res.Body[start:]
  166. } else {
  167. res.Body = _emptyList
  168. }
  169. }
  170. if len(res.Body) == 0 {
  171. res.Body = _emptyList
  172. }
  173. }
  174. return
  175. }
  176. // loadColumnsCache load all columns cache
  177. func (s *Service) loadColumnsCache() {
  178. res, err := s.cdao.Columns(context.TODO())
  179. if err != nil {
  180. log.Error("s.cdao.Columns error(%v)", err)
  181. return
  182. }
  183. tmp := map[string]*card.Column{}
  184. for plat, columns := range res {
  185. for _, column := range columns {
  186. key := fmt.Sprintf(_initColumnKey, plat, column.ID)
  187. tmp[key] = column
  188. }
  189. }
  190. s.columnsCache = tmp
  191. }
  192. // loadColumnListCache
  193. func (s *Service) loadColumnListCache(now time.Time) {
  194. var (
  195. tmp = map[string][]*card.ColumnList{}
  196. )
  197. platColumns, err := s.cdao.ColumnPlatList(context.TODO(), now)
  198. if err != nil {
  199. log.Error("s.cdao.ColumnPlatList error(%v)", err)
  200. return
  201. }
  202. for plat, columns := range platColumns {
  203. for _, column := range columns {
  204. key := fmt.Sprintf(_initColumnListKey, plat, column.Ceid)
  205. tmp[key] = append(tmp[key], column)
  206. }
  207. }
  208. s.cardListCache = tmp
  209. }
  210. // loadNperCache
  211. func (s *Service) loadNperCache(now time.Time) {
  212. hdm, err := s.cdao.ColumnNpers(context.TODO(), now)
  213. if err != nil {
  214. log.Error("s.cdao.ColumnNpers error(%v)", err)
  215. return
  216. }
  217. itm, aids, err := s.cdao.NperContents(context.TODO(), now)
  218. if err != nil {
  219. log.Error("s.cdao.NperContents error(%v)", err)
  220. return
  221. }
  222. tmp, tmpColumns, tmpList := s.mergeCard(context.TODO(), hdm, itm, aids, now)
  223. s.cardCache = tmp
  224. s.columnCache = tmpColumns
  225. s.columnListCache = tmpList
  226. }
  227. // cacheproc load all cache.
  228. func (s *Service) cacheproc() {
  229. for {
  230. time.Sleep(s.tick)
  231. now := time.Now()
  232. s.loadColumnListCache(now)
  233. s.loadColumnsCache()
  234. s.loadNperCache(now)
  235. }
  236. }
  237. // mergeCard
  238. func (s *Service) mergeCard(c context.Context, hdm map[int8][]*card.ColumnNper, itm map[int][]*card.Content, itmaids map[int][]int64, now time.Time) (res map[string][]*daily.Show, columns map[string]*daily.Show, columnList map[string][]*daily.Item) {
  239. var (
  240. dailyMAX = 31
  241. )
  242. res = map[string][]*daily.Show{}
  243. columnList = map[string][]*daily.Item{}
  244. columns = map[string]*daily.Show{}
  245. for plat, hds := range hdm {
  246. for _, hd := range hds {
  247. var (
  248. ok bool
  249. column *card.Column
  250. )
  251. columnskey := fmt.Sprintf(_initColumnKey, plat, hd.ColumnID)
  252. if column, ok = s.columnsCache[columnskey]; !ok {
  253. continue
  254. }
  255. switch column.Type {
  256. case model.GotoDaily:
  257. if dailykey := fmt.Sprintf(_initDailyKey, plat, hd.ColumnID); len(res[dailykey]) > dailyMAX {
  258. continue
  259. }
  260. }
  261. var (
  262. sis []*daily.Item
  263. )
  264. its, ok := itm[hd.ID]
  265. if !ok {
  266. its = []*card.Content{}
  267. }
  268. switch column.Tpl {
  269. case 1, 2:
  270. var tmpItem = map[int64]*daily.Item{}
  271. if aids, ok := itmaids[hd.ID]; ok {
  272. tmpItem = s.fromCardAids(context.TODO(), aids)
  273. }
  274. for _, ci := range its {
  275. si := s.fillCardItem(ci, tmpItem)
  276. if si.Title == "" {
  277. continue
  278. }
  279. if ci.TagID > 0 {
  280. si.TagName, si.TagID = s.fromTagIDByName(c, ci.TagID, now)
  281. }
  282. sis = append(sis, si)
  283. }
  284. }
  285. if len(sis) == 0 {
  286. continue
  287. }
  288. sw := &daily.Show{}
  289. sw.Head = &daily.Head{
  290. ColumnID: hd.ID,
  291. Build: hd.Build,
  292. Condition: hd.Condition,
  293. Plat: hd.Plat,
  294. Desc: hd.Desc,
  295. Type: column.Type,
  296. }
  297. if hd.Cover != "" {
  298. sw.Cover = hd.Cover
  299. }
  300. var key string
  301. switch sw.Head.Type {
  302. case model.GotoDaily:
  303. key = fmt.Sprintf(_initDailyKey, plat, hd.ColumnID)
  304. sw.Head.Title = hd.Name
  305. if len(res[key]) == 0 {
  306. sw.Head.Date = now.Unix()
  307. } else {
  308. sw.Head.Date = int64(hd.NperTime)
  309. }
  310. sw.Body = sis
  311. res[key] = append(res[key], sw)
  312. case model.GotoColumn:
  313. key = fmt.Sprintf(_initDailyKey, plat, hd.ID)
  314. sw.Head.Title = hd.Name
  315. sw.Head.Goto = hd.Goto
  316. sw.Head.Param = hd.Param
  317. sw.Head.URI = hd.URI
  318. columnList[key] = sis
  319. columns[key] = sw
  320. }
  321. }
  322. }
  323. return
  324. }
  325. // fillCardItem
  326. func (s *Service) fillCardItem(csi *card.Content, tsi map[int64]*daily.Item) (si *daily.Item) {
  327. si = &daily.Item{}
  328. switch csi.Type {
  329. case model.CardGotoAv:
  330. si.Goto = model.GotoAv
  331. si.Param = csi.Value
  332. }
  333. si.URI = model.FillURI(si.Goto, si.Param, nil)
  334. if si.Goto == model.GotoAv {
  335. aid, err := strconv.ParseInt(si.Param, 10, 64)
  336. if err != nil {
  337. log.Error("strconv.ParseInt(%s) error(%v)", si.Param, err)
  338. } else {
  339. if it, ok := tsi[aid]; ok {
  340. si = it
  341. if csi.Title != "" {
  342. si.Title = csi.Title
  343. }
  344. } else {
  345. si = &daily.Item{}
  346. }
  347. }
  348. }
  349. return
  350. }
  351. // fromCardAids get Aids.
  352. func (s *Service) fromCardAids(ctx context.Context, aids []int64) (data map[int64]*daily.Item) {
  353. var (
  354. arc *api.Arc
  355. ok bool
  356. )
  357. as, err := s.arc.ArchivesPB(ctx, aids)
  358. if err != nil {
  359. log.Error("s.arc.ArchivesPB(%v) error(%v)", aids, err)
  360. return
  361. }
  362. if len(as) == 0 {
  363. log.Warn("s.arc.ArchivesPB(%v) length is 0", aids)
  364. return
  365. }
  366. data = map[int64]*daily.Item{}
  367. for _, aid := range aids {
  368. if arc, ok = as[aid]; ok {
  369. if !arc.IsNormal() {
  370. continue
  371. }
  372. i := &daily.Item{}
  373. i.FromArchivePB(arc)
  374. data[aid] = i
  375. }
  376. }
  377. return
  378. }
  379. // fromTagIDByName from tag_id by tag_name
  380. func (s *Service) fromTagIDByName(ctx context.Context, tagID int, now time.Time) (tagName string, tagIDInt int64) {
  381. tag, err := s.tag.TagInfo(ctx, 0, tagID, now)
  382. if err != nil {
  383. log.Error("s.tag.TagInfo(%d) error(%v)", tagID, err)
  384. return
  385. }
  386. tagName = tag.Name
  387. tagIDInt = tag.Tid
  388. return
  389. }