h5.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561
  1. package academy
  2. import (
  3. "context"
  4. "sort"
  5. "strconv"
  6. "time"
  7. "go-common/app/interface/main/creative/dao/tool"
  8. "go-common/app/interface/main/creative/model/academy"
  9. "go-common/library/ecode"
  10. "go-common/library/log"
  11. "go-common/library/sync/errgroup"
  12. "go-common/app/service/main/archive/api"
  13. "github.com/davecgh/go-spew/spew"
  14. )
  15. // Tags get all h5 tags.
  16. func (s *Service) Tags(c context.Context) (res []*academy.Tag) {
  17. if v, ok := s.TagsCache[academy.TagClassMap(academy.H5)]; ok {
  18. res = v
  19. }
  20. return
  21. }
  22. // Archives get all h5 archive.
  23. func (s *Service) Archives(c context.Context, aca *academy.EsParam) (res *academy.ArchiveList, err error) {
  24. tids, err := s.webTags(aca.Tid)
  25. if err != nil {
  26. return
  27. }
  28. aca.Tid = tids
  29. res, err = s.ArchivesWithES(c, aca)
  30. return
  31. }
  32. func (s *Service) webTags(tids []int64) (webTIDs []int64, err error) {
  33. var (
  34. lts []*academy.LinkTag
  35. h5TIDs []int64
  36. )
  37. if len(tids) > 0 {
  38. h5TIDs = tids
  39. } else {
  40. tgs := s.Tags(context.Background())
  41. for _, v := range tgs {
  42. h5TIDs = append(h5TIDs, v.ID)
  43. }
  44. }
  45. lts, err = s.aca.LinkTags(context.Background(), h5TIDs)
  46. if err != nil {
  47. return
  48. }
  49. for _, v := range lts {
  50. webTIDs = append(webTIDs, v.LinkID)
  51. }
  52. return
  53. }
  54. // RecommendV2 get recommend archive.
  55. func (s *Service) RecommendV2(c context.Context, mid int64) (res []*academy.RecArcList, err error) {
  56. mainIDMap := make(map[int64]struct{}) //主题课程aid map for rm dup
  57. tgList, err := s.getRecTag(c, mid)
  58. if err != nil {
  59. return
  60. }
  61. log.Info("Recommend mid(%d)|tgList(%s)", mid, spew.Sdump(tgList))
  62. s.setSeed() //init Seed.
  63. res = make([]*academy.RecArcList, 0)
  64. var (
  65. g, _ = errgroup.WithContext(c)
  66. hotItems []*academy.RecArchive
  67. )
  68. g.Go(func() error {
  69. // get hot archives
  70. hotItems, err = s.hotArchives(c)
  71. if err != nil {
  72. log.Error("Recommend s.hotArchives mid(%d)", mid)
  73. return err
  74. }
  75. return nil
  76. })
  77. for _, t := range tgList {
  78. if t == nil {
  79. continue
  80. }
  81. pid, v := t.PID, t.TIDs
  82. if pid == 0 { //主题课程
  83. var ocid int64
  84. if len(v) > 0 {
  85. ocid = v[0]
  86. }
  87. rec := &academy.RecArcList{
  88. TID: ocid,
  89. Items: []*academy.RecArchive{},
  90. }
  91. tg, o := s.OccMapCache[ocid]
  92. if !o || tg == nil {
  93. log.Error("s.OccMapCache ocid(%d) not exist", ocid)
  94. continue
  95. }
  96. rec.Name = tg.Name
  97. items, themeCourseErr := s.themeCourse(c, v)
  98. if themeCourseErr != nil {
  99. return nil, themeCourseErr
  100. }
  101. for _, v := range items {
  102. mainIDMap[v.OID] = struct{}{}
  103. }
  104. rec.Items = items
  105. res = append(res, rec)
  106. } else if tg, ok := s.TagMapCache[pid]; ok { //标签教程
  107. rec := &academy.RecArcList{
  108. TID: pid,
  109. Name: tg.Name,
  110. Items: []*academy.RecArchive{},
  111. }
  112. items, tagCourseErr := s.tagCourse(c, pid, v, mainIDMap)
  113. if tagCourseErr != nil {
  114. return nil, tagCourseErr
  115. }
  116. rec.Items = items
  117. res = append(res, rec)
  118. }
  119. }
  120. if g.Wait() != nil {
  121. log.Error("Recommend s.hotArchives g.Wait() error(%v)", err)
  122. return
  123. }
  124. // add host archives
  125. hotRec := &academy.RecArcList{
  126. TID: 0,
  127. Name: "热门推荐",
  128. Items: hotItems,
  129. }
  130. res = append(res, hotRec)
  131. return
  132. }
  133. func (s *Service) tagCourse(c context.Context, pid int64, v []int64, aidMap map[int64]struct{}) (res []*academy.RecArchive, err error) {
  134. res = make([]*academy.RecArchive, 0)
  135. aca := &academy.EsParam{
  136. Tid: v,
  137. Pn: 1,
  138. Ps: 10,
  139. }
  140. if s.Seed > 0 { //取材创意/视频制作/个人运营 每日0点请求搜索更换时间种子
  141. aca.Seed = s.Seed
  142. }
  143. arcs, err := s.Archives(c, aca)
  144. if err != nil {
  145. log.Error("Recommend s.Archives EsParam(%+v)|error(%v)", aca, err)
  146. return nil, err
  147. }
  148. if arcs == nil {
  149. err = ecode.CreativeAcademyH5RecommendErr
  150. return nil, err
  151. }
  152. var aids []int64
  153. for _, i := range arcs.Items {
  154. // ignore if exist in resource service
  155. if _, exist := s.ResourceMapCache[i.OID]; exist {
  156. continue
  157. }
  158. // ignore if exist in main topic
  159. if _, exist := aidMap[i.OID]; exist {
  160. continue
  161. }
  162. ra := &academy.RecArchive{
  163. OID: i.OID,
  164. MID: i.MID,
  165. Cover: i.Cover,
  166. Title: i.Title,
  167. Business: i.Business,
  168. Duration: i.Duration,
  169. ArcStat: i.ArcStat,
  170. ArtStat: i.ArtStat,
  171. }
  172. res = append(res, ra)
  173. aids = append(aids, i.OID)
  174. }
  175. // add tags
  176. tags, err := s.getTags(c, aids)
  177. if err != nil {
  178. log.Error("tagCourse s.getTags err(%v)", err)
  179. return
  180. }
  181. s.setTags(res, tags)
  182. return
  183. }
  184. func (s *Service) themeCourse(c context.Context, v []int64) (res []*academy.RecArchive, err error) {
  185. res = make([]*academy.RecArchive, 0)
  186. arcs, err := s.ThemeCourse(c, v, []int64{}, []int64{}, 1, 10, false)
  187. if err != nil {
  188. log.Error("Recommend s.ThemeCourse v(%+v)|error(%v)", v, err)
  189. return nil, err
  190. }
  191. if arcs == nil {
  192. err = ecode.CreativeAcademyH5RecommendErr
  193. return nil, err
  194. }
  195. var aids []int64
  196. for _, i := range arcs.Items {
  197. // ignore if exist in resource service
  198. if _, exist := s.ResourceMapCache[i.AID]; exist {
  199. continue
  200. }
  201. ra := &academy.RecArchive{
  202. OID: i.AID,
  203. MID: i.MID,
  204. Cover: i.Cover,
  205. Title: i.Title,
  206. Duration: i.Duration,
  207. ArcStat: i.ArcStat,
  208. Business: academy.BusinessForArchive,
  209. }
  210. res = append(res, ra)
  211. aids = append(aids, i.AID)
  212. }
  213. // add tags
  214. tags, err := s.getTags(c, aids)
  215. if err != nil {
  216. log.Error("themeCourse s.getTags err(%v)", err)
  217. return
  218. }
  219. s.setTags(res, tags)
  220. s.randomForMainCourse(res) //每日0点随机随机稿件列表
  221. if len(s.RecommendArcs) > 0 {
  222. res = s.RecommendArcs
  223. }
  224. return
  225. }
  226. func (s *Service) getRecTag(c context.Context, mid int64) (res []*academy.RecConf, err error) {
  227. var tyID int64
  228. if mid > 0 {
  229. tyID, err = s.getFavType(c, mid) //获取推荐分区id
  230. if err != nil {
  231. log.Error("getFavType mid(%d)|error(%v)", mid, err)
  232. } else {
  233. log.Info("getFavType mid(%d)|tyID(%d)", mid, tyID)
  234. }
  235. }
  236. if s.c == nil || s.c.AcaRecommend == nil {
  237. log.Error("getRecTag get conf error mid(%d)", mid)
  238. return
  239. }
  240. rec := s.c.AcaRecommend.Recommend
  241. //按 主题课程-取材创意-视频制作-个人运营 排序
  242. var rec1, rec2, rec3, rec4 *academy.RecConf
  243. res = make([]*academy.RecConf, 0, 4)
  244. //主题课程
  245. if rec.Course != nil {
  246. course := rec.Course
  247. rec1 = &academy.RecConf{PID: course.ID}
  248. if tyID != 0 {
  249. if tool.ElementInSlice(tyID, course.Shoot.Val) { //如果最近投稿分区命中配置的分区,则设置当前一级分类下面的标签为最近投稿分区对应的二级标签目录
  250. rec1.TIDs = course.Shoot.Key
  251. } else if tool.ElementInSlice(tyID, course.Scene.Val) {
  252. rec1.TIDs = course.Scene.Key
  253. } else if tool.ElementInSlice(tyID, course.Edit.Val) {
  254. rec1.TIDs = course.Edit.Key
  255. } else if tool.ElementInSlice(tyID, course.Mmd.Val) {
  256. rec1.TIDs = course.Mmd.Key
  257. } else if tool.ElementInSlice(tyID, course.Sing.Val) {
  258. rec1.TIDs = course.Sing.Key
  259. } else if tool.ElementInSlice(tyID, course.Bang.Val) {
  260. rec1.TIDs = course.Bang.Key
  261. }
  262. } else {
  263. rec1.TIDs = course.Other.Key
  264. }
  265. res = append(res, rec1)
  266. } else {
  267. log.Error("getRecTag get cousre conf mid(%d)", mid)
  268. }
  269. //取材创意
  270. if rec.Drawn != nil {
  271. drawn := rec.Drawn
  272. rec2 = &academy.RecConf{PID: drawn.ID}
  273. if tyID != 0 {
  274. if tool.ElementInSlice(tyID, drawn.MobilePlan.Val) { //如果最近投稿分区命中配置的分区,则设置当前一级分类下面的标签为最近投稿分区对应的二级标签目录
  275. rec2.TIDs = drawn.MobilePlan.Key
  276. } else if tool.ElementInSlice(tyID, drawn.ScreenPlan.Val) {
  277. rec2.TIDs = drawn.ScreenPlan.Key
  278. } else if tool.ElementInSlice(tyID, drawn.RecordPlan.Val) {
  279. rec2.TIDs = drawn.RecordPlan.Key
  280. }
  281. } else {
  282. rec2.TIDs = drawn.Other.Key
  283. }
  284. res = append(res, rec2)
  285. } else {
  286. log.Error("getRecTag get drawn conf mid(%d)", mid)
  287. }
  288. //视频制作
  289. if rec.Video != nil {
  290. video := rec.Video
  291. rec3 = &academy.RecConf{PID: video.ID}
  292. if tyID != 0 {
  293. if tool.ElementInSlice(tyID, video.MobileMake.Val) { //如果最近投稿分区命中配置的分区,则设置当前一级分类下面的标签为最近投稿分区对应的二级标签目录
  294. rec3.TIDs = video.MobileMake.Key
  295. } else if tool.ElementInSlice(tyID, video.AudioEdit.Val) {
  296. rec3.TIDs = video.AudioEdit.Key
  297. } else if tool.ElementInSlice(tyID, video.EditCompose.Val) {
  298. rec3.TIDs = video.EditCompose.Key
  299. }
  300. } else {
  301. rec3.TIDs = video.Other.Key
  302. }
  303. res = append(res, rec3)
  304. } else {
  305. log.Error("getRecTag get video conf mid(%d)", mid)
  306. }
  307. //个人运营
  308. if rec.Person != nil {
  309. person := rec.Person
  310. rec4 = &academy.RecConf{PID: person.ID, TIDs: person.Other.Key}
  311. res = append(res, rec4)
  312. } else {
  313. log.Error("getRecTag get person conf mid(%d)", mid)
  314. }
  315. return
  316. }
  317. func (s *Service) getFavType(c context.Context, mid int64) (tyID int64, err error) { //获取最近投稿的一个分区
  318. tys, err := s.arc.FavTypes(c, mid)
  319. if err != nil {
  320. log.Error("s.arc.FavTypes mid(%d)|error(%v)", mid, err)
  321. return
  322. }
  323. if len(tys) == 0 {
  324. return 0, nil
  325. }
  326. type kv struct {
  327. id int64
  328. ptime int64
  329. }
  330. var tps []*kv
  331. for id, t := range tys {
  332. tid, err := strconv.ParseInt(id, 10, 64)
  333. if err != nil {
  334. return 0, err
  335. }
  336. tps = append(tps, &kv{tid, t})
  337. }
  338. sort.Slice(tps, func(i, j int) bool {
  339. return tps[i].ptime > tps[j].ptime
  340. })
  341. if len(tps) > 0 && tps[0] != nil {
  342. tyID = tps[0].id
  343. }
  344. return
  345. }
  346. //randomForMainCourse 主题课程每日0点重新随机排序
  347. func (s *Service) randomForMainCourse(arc []*academy.RecArchive) {
  348. count := len(arc)
  349. if count == 0 {
  350. return
  351. }
  352. keys := tool.RandomSliceKeys(0, count, count, s.Seed)
  353. res := make([]*academy.RecArchive, 0, count)
  354. for _, k := range keys {
  355. res = append(res, arc[k])
  356. }
  357. if len(res) > 0 { //获取随机排序的稿件列表
  358. s.RecommendArcs = res
  359. }
  360. log.Info("randomRecommend s.RecommendArcs (%s)", spew.Sdump(s.RecommendArcs))
  361. }
  362. func (s *Service) setSeed() {
  363. now := time.Now()
  364. last := now
  365. next := now.Add(time.Hour * 24)
  366. last = time.Date(last.Year(), last.Month(), last.Day(), 0, 0, 0, 0, last.Location()) //昨日0点
  367. next = time.Date(next.Year(), next.Month(), next.Day(), 0, 0, 0, 0, next.Location()) //明日0点
  368. if now.Unix() > last.Unix() && now.Unix() < next.Unix() {
  369. s.Seed = last.Unix() //set last seed
  370. } else {
  371. s.Seed = next.Unix() //set next seed
  372. }
  373. log.Info("setSeed s.Seed (%d)", s.Seed)
  374. }
  375. func (s *Service) getTags(c context.Context, aids []int64) (res map[int64]map[string][]*academy.Tag, err error) {
  376. if len(aids) == 0 {
  377. log.Error("getTags len(aids) == 0")
  378. return
  379. }
  380. aidTIDsMap, err := s.aca.ArchiveTagsByOids(c, aids)
  381. if err != nil {
  382. log.Error("getTags s.aca.ArchiveTagsByOids aids(%+v)", aids)
  383. return
  384. }
  385. if len(aidTIDsMap) == 0 {
  386. log.Error("getTags len(aidTIDsMap) == 0 | aids(%+v)", aids)
  387. return
  388. }
  389. res, err = s.bindTags(c, aidTIDsMap)
  390. if err != nil {
  391. log.Error("getTags s.bindTags | err(%v)", err)
  392. return
  393. }
  394. return
  395. }
  396. func (s *Service) setTags(x interface{}, tags map[int64]map[string][]*academy.Tag) {
  397. switch arcs := x.(type) {
  398. case []*academy.RecArchive:
  399. for _, v := range arcs {
  400. if v != nil {
  401. if tag, ok := tags[v.OID]; ok {
  402. v.Tags = tag
  403. }
  404. }
  405. }
  406. case []*academy.ArcMeta:
  407. for _, v := range arcs {
  408. if v != nil {
  409. if tag, ok := tags[v.AID]; ok {
  410. v.Tags = tag
  411. }
  412. }
  413. }
  414. }
  415. }
  416. // HotArchives get host archives
  417. func (s *Service) HotArchives(c context.Context, oids []int64) (res []*academy.ArchiveMeta, err error) {
  418. res = make([]*academy.ArchiveMeta, 0)
  419. if len(oids) == 0 {
  420. log.Error("HotArchives len(oids) == 0")
  421. return
  422. }
  423. var (
  424. g, _ = errgroup.WithContext(c)
  425. arcs map[int64]*api.Arc
  426. st map[int64]*api.Stat
  427. )
  428. g.Go(func() error {
  429. arcs, err = s.arc.Archives(c, oids, "")
  430. if err != nil {
  431. log.Error("HotArchives s.arc.Archives oids(%+v)| error(%v)", oids, err)
  432. return err
  433. }
  434. st, err = s.arc.Stats(c, oids, "")
  435. if err != nil {
  436. log.Error("HotArchives s.arc.Stats oids(%+v)| error(%v)", oids, err)
  437. return err
  438. }
  439. return nil
  440. })
  441. if g.Wait() != nil {
  442. log.Error("HotArchives g.Wait() error(%v)", err)
  443. return
  444. }
  445. for _, oid := range oids {
  446. a := &academy.ArchiveMeta{
  447. OID: oid,
  448. }
  449. a = bindArchiveInfo(oid, arcs, a)
  450. if t, ok := st[oid]; ok {
  451. a.ArcStat = t
  452. } else {
  453. a.ArcStat = &api.Stat{}
  454. }
  455. res = append(res, a)
  456. }
  457. return
  458. }
  459. func (s *Service) hotArchives(c context.Context) (res []*academy.RecArchive, err error) {
  460. if len(s.ResourceMapCache) == 0 {
  461. log.Error("hotArchives len(oids) == 0 | ResourceMapCache(%+v)", s.ResourceMapCache)
  462. return
  463. }
  464. res = make([]*academy.RecArchive, 0)
  465. var oids []int64
  466. for k := range s.ResourceMapCache {
  467. oids = append(oids, k)
  468. }
  469. arcs, err := s.HotArchives(c, oids)
  470. if err != nil {
  471. log.Error("hotArchives s.HotArchives oids(%+v) | error(%v)", oids, err)
  472. return
  473. }
  474. var aids []int64
  475. for _, v := range arcs {
  476. ra := &academy.RecArchive{
  477. OID: v.OID,
  478. MID: v.MID,
  479. Cover: v.Cover,
  480. Title: v.Title,
  481. Business: 1, //热门推荐默认为视频
  482. Duration: v.Duration,
  483. ArcStat: v.ArcStat,
  484. ArtStat: v.ArtStat,
  485. }
  486. res = append(res, ra)
  487. aids = append(aids, v.OID)
  488. }
  489. //add tags
  490. tags, err := s.getTags(c, aids)
  491. if err != nil {
  492. log.Error("hotArchives s.getTags err(%v)", err)
  493. return
  494. }
  495. s.setTags(res, tags)
  496. return
  497. }