service.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501
  1. package weeklyhonor
  2. import (
  3. "context"
  4. "crypto/md5"
  5. "encoding/hex"
  6. "fmt"
  7. "time"
  8. "go-common/app/interface/main/creative/conf"
  9. "go-common/app/interface/main/creative/dao/account"
  10. "go-common/app/interface/main/creative/dao/archive"
  11. "go-common/app/interface/main/creative/dao/up"
  12. "go-common/app/interface/main/creative/dao/weeklyhonor"
  13. model "go-common/app/interface/main/creative/model/weeklyhonor"
  14. "go-common/app/interface/main/creative/service"
  15. accmdl "go-common/app/service/main/account/model"
  16. "go-common/app/service/main/archive/api"
  17. upmdl "go-common/app/service/main/up/model"
  18. "go-common/library/cache/memcache"
  19. "go-common/library/ecode"
  20. "go-common/library/log"
  21. "go-common/library/sync/errgroup"
  22. "go-common/library/sync/pipeline/fanout"
  23. xtime "go-common/library/time"
  24. )
  25. const (
  26. layout = "20060102"
  27. )
  28. // Service struct.
  29. type Service struct {
  30. c *conf.Config
  31. honDao *weeklyhonor.Dao
  32. arc *archive.Dao
  33. acc *account.Dao
  34. up *up.Dao
  35. // cache chan
  36. cache *fanout.Fanout
  37. honorMap map[int][]*model.HonorWord
  38. }
  39. // New get service.
  40. func New(c *conf.Config, rpcdaos *service.RPCDaos) *Service {
  41. s := &Service{
  42. c: c,
  43. honDao: weeklyhonor.New(c),
  44. arc: rpcdaos.Arc,
  45. acc: rpcdaos.Acc,
  46. up: rpcdaos.Up,
  47. // cache
  48. cache: fanout.New("cache"),
  49. }
  50. s.honorMap = model.HMap()
  51. return s
  52. }
  53. // Ping service.
  54. func (s *Service) Ping(c context.Context) (err error) {
  55. if err = s.honDao.Ping(c); err != nil {
  56. log.Error("s.honor.Ping err(%v)", err)
  57. }
  58. return
  59. }
  60. // Close dao.
  61. func (s *Service) Close() {
  62. s.honDao.Close()
  63. }
  64. // ChangeSubState change subscribe state
  65. func (s *Service) ChangeSubState(c context.Context, mid int64, state uint8) (err error) {
  66. err = s.honDao.ChangeUpSwitch(c, mid, state)
  67. if err != nil {
  68. log.Error("s.honDao.ChangeUpSwitch mid(%d) state(%d) err(%v)", mid, state, err)
  69. }
  70. return
  71. }
  72. // WeeklyHonor .
  73. func (s *Service) WeeklyHonor(c context.Context, mid, uid int64, token string) (h *model.Honor, err error) {
  74. var upMid = mid
  75. h = &model.Honor{}
  76. if uid != 0 && uid != upMid {
  77. if token != s.genToken(uid) {
  78. return nil, ecode.RequestErr
  79. }
  80. upMid = uid
  81. }
  82. h.MID = upMid
  83. var upInfo *upmdl.UpInfo
  84. upInfo, err = s.up.UpInfo(c, upMid, 0, "")
  85. if err != nil {
  86. log.Error("s.up.UpInfo(%d) error(%v)", upMid, err)
  87. return
  88. }
  89. if upInfo.IsAuthor != 1 {
  90. return nil, ecode.CreativeNotUper
  91. }
  92. if upMid == mid {
  93. h.SubState, err = s.honDao.GetUpSwitch(c, upMid)
  94. if err != nil {
  95. log.Error("s.honDao.GetUpSwitch upMid(%d),err(%v)", upMid, err)
  96. }
  97. go s.addHonorClickCount(context.Background(), mid)
  98. }
  99. var timeUp bool
  100. h.DateBegin, h.DateEnd, timeUp = honTimeFrame()
  101. endStr := time.Unix(int64(h.DateEnd), 0).Format(layout)
  102. h.ShareToken = s.genToken(upMid)
  103. hs, err := s.honorStat(c, upMid, endStr)
  104. if hs == nil || err != nil || s.c.HonorDegradeSwitch {
  105. return s.degrade(c, h)
  106. }
  107. hl, err := s.honDao.HonorMC(c, h.MID, endStr)
  108. if err != nil {
  109. log.Error("s.honDao.HonorMC() error(%v)", err)
  110. return s.degrade(c, h)
  111. }
  112. if hl != nil {
  113. h.HID = hl.HID
  114. h.HonorCount = hl.Count
  115. } else {
  116. h, err = s.genHonor(c, h, endStr, timeUp, hs)
  117. if err != nil {
  118. return
  119. }
  120. }
  121. if h.HID == 0 {
  122. return s.degrade(c, h)
  123. }
  124. if err = s.rpcFill(c, hs, h); err != nil {
  125. return
  126. }
  127. s.wordFill(hs, h)
  128. h.RiseStage = s.stars(hs)
  129. return
  130. }
  131. func (s *Service) addHonorClickCount(c context.Context, mid int64) {
  132. err := s.honDao.ClickMC(c, mid)
  133. if err != memcache.ErrNotFound {
  134. if err != nil {
  135. log.Error("s.honDao.ClickMC mid(%d) err(%+v)", mid, err)
  136. }
  137. return
  138. }
  139. err = s.honDao.SetClickMC(c, mid)
  140. if err != nil {
  141. log.Error("s.honDao.SetClickMC mid(%d) err(%+v)", mid, err)
  142. return
  143. }
  144. err = s.honDao.UpsertClickCount(c, mid)
  145. if err != nil {
  146. log.Error("failed to add honor click count,mid(%d) err(%+v)", mid, err)
  147. }
  148. }
  149. func (s *Service) honorStat(c context.Context, mid int64, date string) (hs *model.HonorStat, err error) {
  150. hs, err = s.honDao.StatMC(c, mid, date)
  151. if hs != nil && err == nil {
  152. return
  153. }
  154. for i := 0; i < 3; i++ {
  155. hs, err = s.honDao.HonorStat(c, mid, date)
  156. if err != nil {
  157. log.Error("s.honDao.HonorStat(%d,%v) error(%v)", mid, date, err)
  158. continue
  159. }
  160. if hs != nil {
  161. break
  162. }
  163. }
  164. _ = s.cache.Do(c, func(c context.Context) {
  165. _ = s.honDao.SetStatMC(c, mid, date, hs)
  166. })
  167. return
  168. }
  169. func (s *Service) degrade(c context.Context, h *model.Honor) (*model.Honor, error) {
  170. h.Word = s.honorMap[58][0].Word
  171. h.Text = s.honorMap[58][0].Text
  172. h.Priority = s.honorMap[58][0].Priority
  173. us, err := s.acc.Infos(c, []int64{h.MID}, "")
  174. if err != nil {
  175. log.Error("s.acc.Infos(%v) error(%v)", h.MID, err)
  176. return nil, err
  177. }
  178. if u, ok := us[h.MID]; ok {
  179. h.Uname = u.Name
  180. h.Face = u.Face
  181. }
  182. return h, nil
  183. }
  184. func (s *Service) wordFill(hs *model.HonorStat, h *model.Honor) {
  185. hw := s.wordFormat(h, hs)
  186. h.Word = hw.Word
  187. h.Text = hw.Text
  188. h.Desc = hw.Desc
  189. h.Priority = hw.Priority
  190. }
  191. func (s *Service) wordFormat(h *model.Honor, stat *model.HonorStat) *model.HonorWord {
  192. hid := h.HID
  193. uname := h.Uname
  194. hm := s.honorMap
  195. hw := new(model.HonorWord)
  196. if _, ok := hm[hid]; !ok {
  197. return hw
  198. }
  199. hwfmt := s.getHWFmt(h)
  200. hw.Word = hwfmt.Word
  201. hw.Text = hwfmt.Text
  202. hw.Desc = hwfmt.Desc
  203. hw.Priority = hwfmt.Priority
  204. switch hid {
  205. case 2, 3, 4, 59:
  206. hw.Desc = fmt.Sprintf(hwfmt.Desc, stat.Rank0)
  207. case 5:
  208. var str string
  209. if stat.Play/100000000 > 0 {
  210. str = fmt.Sprintf("%d个亿", stat.Play/100000000)
  211. } else {
  212. str = fmt.Sprintf("%d千万", stat.Play/10000000)
  213. }
  214. hw.Desc = fmt.Sprintf(hwfmt.Desc, str)
  215. case 6:
  216. hw.Text = fmt.Sprintf(hwfmt.Text, uname)
  217. hw.Desc = fmt.Sprintf(hwfmt.Desc, stat.Fans/1000000)
  218. case 8:
  219. _, partion, rank := stat.PartionRank()
  220. hw.Desc = fmt.Sprintf(hwfmt.Desc, partion, rank)
  221. case 9:
  222. num := stat.Play / 100000
  223. if stat.Play/1000000 > 0 {
  224. num = stat.Play / 1000000 * 10
  225. }
  226. hw.Text = fmt.Sprintf(hwfmt.Text, num*10*2)
  227. hw.Desc = fmt.Sprintf(hwfmt.Desc, num*10)
  228. case 10:
  229. hw.Desc = fmt.Sprintf(hwfmt.Desc, stat.Fans/10000)
  230. case 12, 30:
  231. hw.Text = fmt.Sprintf(hwfmt.Text, uname)
  232. case 17:
  233. hw.Text = fmt.Sprintf(hwfmt.Text, stat.Play/10000*2)
  234. hw.Desc = fmt.Sprintf(hwfmt.Desc, stat.Play/10000)
  235. case 18:
  236. hw.Desc = fmt.Sprintf(hwfmt.Desc, stat.Fans/10000)
  237. case 26:
  238. hw.Desc = fmt.Sprintf(hwfmt.Desc, stat.Play/1000)
  239. case 27:
  240. hw.Desc = fmt.Sprintf(hwfmt.Desc, stat.Fans/1000)
  241. }
  242. return hw
  243. }
  244. func (s *Service) stars(hs *model.HonorStat) *model.RiseStage {
  245. _, stars := hs.PrioritySR()
  246. _, starsR := hs.PriorityR()
  247. stars = s.outputBig(stars, starsR)
  248. _, starsA := hs.PriorityA()
  249. stars = s.outputBig(stars, starsA)
  250. _, starsB := hs.PriorityB()
  251. stars = s.outputBig(stars, starsB)
  252. _, starsC := hs.PriorityC()
  253. stars = s.outputBig(stars, starsC)
  254. return stars
  255. }
  256. func (s *Service) outputBig(a, b *model.RiseStage) *model.RiseStage {
  257. if b.Coin > a.Coin {
  258. a.Coin = b.Coin
  259. }
  260. if b.Fans > a.Fans {
  261. a.Fans = b.Fans
  262. }
  263. if b.Like > a.Like {
  264. a.Like = b.Like
  265. }
  266. if b.Play > a.Play {
  267. a.Play = b.Play
  268. }
  269. if b.Share > a.Share {
  270. a.Share = b.Share
  271. }
  272. return a
  273. }
  274. // honTimeFrame get honor time frame return start=last week Sun. end= this week Sat.
  275. func honTimeFrame() (start, end xtime.Time, timeUp bool) {
  276. lastSun := model.LatestSunday()
  277. // saturday
  278. endTime := lastSun.AddDate(0, 0, -1)
  279. timeUp = time.Now().Unix() > lastSun.Unix()+18*3600
  280. if !timeUp {
  281. endTime = lastSun.AddDate(0, 0, -8)
  282. }
  283. // the week's (end's week-1) sunday
  284. start = xtime.Time(endTime.AddDate(0, 0, -6).Unix())
  285. end = xtime.Time(endTime.Unix())
  286. return
  287. }
  288. // corHW get correspond HonorWord fmt
  289. func (s *Service) getHWFmt(h *model.Honor) (hw *model.HonorWord) {
  290. hm := s.honorMap
  291. hws := hm[h.HID]
  292. if len(hws) == 1 {
  293. hw = hws[0]
  294. return
  295. }
  296. var (
  297. l = 0
  298. r = len(hws) - 1
  299. m = (l + r) / 2
  300. )
  301. if h.DateEnd > hws[r].Start {
  302. hw = hws[r]
  303. return
  304. }
  305. if h.DateEnd <= hws[l].End {
  306. hw = hws[l]
  307. return
  308. }
  309. for l < r {
  310. if h.DateEnd > hws[m].Start && h.DateEnd <= hws[m].End {
  311. break
  312. } else if h.DateEnd <= hws[m].Start {
  313. r = m - 1
  314. } else {
  315. l = m + 1
  316. }
  317. m = (l + r) / 2
  318. }
  319. hw = hws[m]
  320. return
  321. }
  322. func (s *Service) genHonor(c context.Context, h *model.Honor, endStr string, timeUp bool, hs *model.HonorStat) (*model.Honor, error) {
  323. hls, err := s.honDao.HonorLogs(c, h.MID)
  324. end := h.DateEnd
  325. if err != nil {
  326. log.Error("s.honDao.HonorLogs(%d) error(%v)", h.MID, err)
  327. return s.degrade(c, h)
  328. }
  329. var lastHid int
  330. for _, v := range hls {
  331. if int64(v.MTime) < int64(end) && int64(v.MTime) > int64(end)-7*24*3600 {
  332. lastHid = v.HID
  333. }
  334. }
  335. h.HID = hs.GenHonor(h.MID, lastHid)
  336. if h.HID == 0 {
  337. return s.degrade(c, h)
  338. }
  339. var needUpdate bool
  340. if v, ok := hls[h.HID]; !ok {
  341. needUpdate = true
  342. h.HonorCount = 1
  343. } else {
  344. h.HonorCount = v.Count
  345. if int64(v.MTime) < int64(end) {
  346. needUpdate = true
  347. h.HonorCount = v.Count + 1
  348. }
  349. }
  350. if !timeUp {
  351. return h, nil
  352. }
  353. if needUpdate {
  354. _ = s.cache.Do(c, func(c context.Context) {
  355. if res, err1 := s.honDao.HonorMC(c, h.MID, endStr); err1 != nil || res != nil {
  356. return
  357. }
  358. if err1 := s.honDao.UpsertCount(c, h.MID, h.HID); err1 != nil {
  359. log.Error("s.honDao.UpsertCount() error(%v)", err1)
  360. return
  361. }
  362. hl := &model.HonorLog{
  363. MID: h.MID,
  364. HID: h.HID,
  365. Count: h.HonorCount,
  366. }
  367. if err1 := s.honDao.SetHonorMC(c, h.MID, endStr, hl); err1 != nil {
  368. log.Error("s.honDao.SetHonorMC(%d,%s,%v) error(%v)", h.MID, endStr, hl, err1)
  369. }
  370. })
  371. } else {
  372. hl := &model.HonorLog{
  373. MID: h.MID,
  374. HID: h.HID,
  375. Count: h.HonorCount,
  376. }
  377. if err1 := s.honDao.SetHonorMC(c, h.MID, endStr, hl); err1 != nil {
  378. log.Error("s.honDao.SetHonorMC(%d,%s,%v) error(%v)", h.MID, endStr, hl, err1)
  379. }
  380. }
  381. return h, nil
  382. }
  383. func (s *Service) rpcFill(c context.Context, hs *model.HonorStat, h *model.Honor) error {
  384. mids := []int64{h.MID}
  385. h.LoveFans = make([]*accmdl.Info, 0)
  386. h.PlayFans = make([]*accmdl.Info, 0)
  387. loveFans := make([]int64, 0)
  388. playFans := make([]int64, 0)
  389. aids := make([]int64, 0)
  390. // users
  391. if hs.Act1 != 0 {
  392. loveFans = append(loveFans, int64(hs.Act1))
  393. }
  394. if hs.Act2 != 0 {
  395. loveFans = append(loveFans, int64(hs.Act2))
  396. }
  397. if hs.Act3 != 0 {
  398. loveFans = append(loveFans, int64(hs.Act3))
  399. }
  400. mids = append(mids, loveFans...)
  401. if hs.Dr1 != 0 {
  402. playFans = append(playFans, int64(hs.Dr1))
  403. }
  404. if hs.Dr2 != 0 {
  405. playFans = append(playFans, int64(hs.Dr2))
  406. }
  407. if hs.Dr3 != 0 {
  408. playFans = append(playFans, int64(hs.Dr3))
  409. }
  410. mids = append(mids, playFans...)
  411. // arcs
  412. if hs.HottestAvNew != 0 {
  413. aids = append(aids, int64(hs.HottestAvNew))
  414. }
  415. if hs.HottestAvInc != 0 {
  416. aids = append(aids, int64(hs.HottestAvInc))
  417. }
  418. if hs.HottestAvAll != 0 {
  419. aids = append(aids, int64(hs.HottestAvAll))
  420. }
  421. g := new(errgroup.Group)
  422. var (
  423. us map[int64]*accmdl.Info
  424. arcs map[int64]*api.Arc
  425. )
  426. if len(aids) > 0 {
  427. g.Go(func() (err error) {
  428. arcs, err = s.arc.Archives(c, aids, "")
  429. if err != nil {
  430. log.Error("s.arc.Archives(%v) error(%v)", aids, err)
  431. return err
  432. }
  433. return nil
  434. })
  435. }
  436. g.Go(func() (err error) {
  437. us, err = s.acc.Infos(c, mids, "")
  438. if err != nil {
  439. log.Error("s.acc.Infos(%v) error(%v)", mids, err)
  440. return err
  441. }
  442. return nil
  443. })
  444. err := g.Wait()
  445. if u, ok := us[h.MID]; ok {
  446. h.Uname = u.Name
  447. h.Face = u.Face
  448. }
  449. for _, uid := range loveFans {
  450. if u, ok := us[uid]; ok {
  451. h.LoveFans = append(h.LoveFans, u)
  452. }
  453. }
  454. for _, uid := range playFans {
  455. if u, ok := us[uid]; ok {
  456. h.PlayFans = append(h.PlayFans, u)
  457. }
  458. }
  459. if hs.HottestAvInc != 0 {
  460. if arc, ok := arcs[int64(hs.HottestAvInc)]; ok {
  461. h.NewArchive = arc
  462. }
  463. }
  464. if hs.HottestAvNew != 0 {
  465. if arc, ok := arcs[int64(hs.HottestAvNew)]; ok {
  466. h.NewArchive = arc
  467. }
  468. }
  469. if hs.HottestAvAll != 0 {
  470. if arc, ok := arcs[int64(hs.HottestAvAll)]; ok {
  471. h.HotArchive = arc
  472. }
  473. }
  474. return err
  475. }
  476. func (s *Service) genToken(mid int64) string {
  477. bs := []byte(fmt.Sprintf("bili%d", mid*3+223333))
  478. hs := md5.Sum(bs)
  479. return hex.EncodeToString(hs[:])
  480. }