list.go 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909
  1. package service
  2. import (
  3. "context"
  4. "encoding/json"
  5. "sort"
  6. "strconv"
  7. model "go-common/app/interface/main/reply/model/reply"
  8. accmdl "go-common/app/service/main/account/api"
  9. "go-common/library/ecode"
  10. "go-common/library/log"
  11. "go-common/library/net/metadata"
  12. "go-common/library/sync/errgroup.v2"
  13. )
  14. const (
  15. // hot count web
  16. _webHotCnt = 3
  17. // 热评列表中至少点赞数
  18. _hotLikes = 3
  19. // 取5条,因为有可能管理员置顶和up置顶同时存在
  20. _hotFilters = 5
  21. )
  22. func withinFloor(rpIDs []int64, rpID int64, pn, ps int, asc bool) bool {
  23. if len(rpIDs) == 0 || (pn == 1 && len(rpIDs) < ps) {
  24. return true
  25. }
  26. first := rpIDs[0]
  27. last := rpIDs[len(rpIDs)-1]
  28. if asc {
  29. if (first < rpID && last > rpID) || (len(rpIDs) < ps && last < rpID) {
  30. return true
  31. }
  32. } else {
  33. if (pn == 1 && first < rpID) || (first > rpID && last < rpID) || (len(rpIDs) < ps && last > rpID) {
  34. return true
  35. }
  36. }
  37. return false
  38. }
  39. // SecondReplies return second replies.
  40. func (s *Service) SecondReplies(c context.Context, mid, oid, rootID, jumpID int64, tp int8, pn, ps int, escape bool) (seconds []*model.Reply, root *model.Reply, upMid int64, toPn int, err error) {
  41. var (
  42. ok bool
  43. sub *model.Subject
  44. jump *model.Reply
  45. )
  46. if !model.LegalSubjectType(tp) {
  47. err = ecode.ReplyIllegalSubType
  48. return
  49. }
  50. if sub, err = s.Subject(c, oid, tp); err != nil {
  51. return
  52. }
  53. // jump to the child reply page list
  54. if jumpID > 0 {
  55. if jump, err = s.ReplyContent(c, oid, jumpID, tp); err != nil {
  56. return
  57. }
  58. if jump.Root == 0 {
  59. root = jump
  60. pn = 1
  61. } else {
  62. if root, err = s.ReplyContent(c, oid, jump.Root, tp); err != nil {
  63. return
  64. }
  65. if pos := s.getReplyPosByRoot(c, root, jump); pos > ps {
  66. pn = (pos-1)/ps + 1
  67. } else {
  68. pn = 1
  69. }
  70. }
  71. } else {
  72. if root, err = s.ReplyContent(c, oid, rootID, tp); err != nil {
  73. return
  74. }
  75. if root.Root != 0 {
  76. if root, err = s.ReplyContent(c, oid, root.Root, tp); err != nil {
  77. return
  78. }
  79. }
  80. }
  81. if root.IsDeleted() {
  82. err = ecode.ReplyDeleted
  83. return
  84. }
  85. upMid = sub.Mid
  86. toPn = pn
  87. // get reply second reply content
  88. rootMap := make(map[int64]*model.Reply, 1)
  89. rootMap[root.RpID] = root
  90. secondMap, _, _ := s.secondReplies(c, sub, rootMap, mid, pn, ps)
  91. if seconds, ok = secondMap[root.RpID]; !ok {
  92. seconds = _emptyReplies
  93. }
  94. // get reply dependency info
  95. rs := make([]*model.Reply, 0, len(seconds)+1)
  96. rs = append(rs, root)
  97. rs = append(rs, seconds...)
  98. if err = s.buildReply(c, sub, rs, mid, escape); err != nil {
  99. return
  100. }
  101. return
  102. }
  103. // JumpReplies jump to page by reply id.
  104. func (s *Service) JumpReplies(c context.Context, mid, oid, rpID int64, tp int8, ps, sndPs int, escape bool) (roots, hots []*model.Reply, topAdmin, topUpper *model.Reply, sub *model.Subject, pn, sndPn, total int, err error) {
  105. var (
  106. rootPos, sndPos int
  107. rootRp, rp *model.Reply
  108. fixedSeconds []*model.Reply
  109. )
  110. if !model.LegalSubjectType(tp) {
  111. err = ecode.ReplyIllegalSubType
  112. return
  113. }
  114. if sub, err = s.Subject(c, oid, tp); err != nil {
  115. return
  116. }
  117. if rp, err = s.ReplyContent(c, oid, rpID, tp); err != nil {
  118. return
  119. }
  120. if rp.Root == 0 && rp.Parent == 0 {
  121. rootPos = s.getReplyPos(c, sub, rp)
  122. } else {
  123. if rootRp, err = s.ReplyContent(c, oid, rp.Root, tp); err != nil {
  124. return
  125. }
  126. rootPos = s.getReplyPos(c, sub, rootRp)
  127. sndPos = s.getReplyPosByRoot(c, rootRp, rp)
  128. }
  129. // root page number
  130. pn = (rootPos-1)/ps + 1
  131. // second page number
  132. if sndPos > sndPs {
  133. sndPn = (sndPos-1)/sndPs + 1
  134. } else {
  135. sndPn = 1
  136. }
  137. // get reply content
  138. roots, seconds, total, err := s.rootReplies(c, sub, mid, model.SortByFloor, pn, ps, 1, sndPs)
  139. if err != nil {
  140. return
  141. }
  142. if rootRp != nil && rootRp.RCount > 0 {
  143. if fixedSeconds, err = s.repliesByRoot(c, oid, rootRp.RpID, tp, sndPn, sndPs); err != nil {
  144. return
  145. }
  146. for _, rp := range roots {
  147. if rp.RpID == rootRp.RpID {
  148. rp.Replies = fixedSeconds
  149. }
  150. }
  151. }
  152. // top and hots
  153. topAdmin, topUpper, hots, hseconds, err := s.topAndHots(c, sub, mid, true, true)
  154. if err != nil {
  155. log.Error("s.topAndHots(%d,%d,%d) error(%v)", oid, tp, mid, err)
  156. err = nil // degrade
  157. }
  158. rs := make([]*model.Reply, 0, len(roots)+len(seconds)+len(hseconds)+len(fixedSeconds)+2)
  159. rs = append(rs, roots...)
  160. rs = append(rs, seconds...)
  161. rs = append(rs, hseconds...)
  162. rs = append(rs, hots...)
  163. rs = append(rs, fixedSeconds...)
  164. if topAdmin != nil {
  165. rs = append(rs, topAdmin)
  166. }
  167. if topUpper != nil {
  168. rs = append(rs, topUpper)
  169. }
  170. if err = s.buildReply(c, sub, rs, mid, escape); err != nil {
  171. return
  172. }
  173. return
  174. }
  175. // RootReplies return a page list of reply.
  176. func (s *Service) RootReplies(c context.Context, params *model.PageParams) (page *model.PageResult, err error) {
  177. if !model.LegalSubjectType(params.Type) {
  178. err = ecode.ReplyIllegalSubType
  179. return
  180. }
  181. sub, err := s.Subject(c, params.Oid, params.Type)
  182. if err != nil {
  183. return
  184. }
  185. topAdmin, topUpper, hots, hseconds, err := s.topAndHots(c, sub, params.Mid, params.NeedHot, params.NeedSecond)
  186. if err != nil {
  187. log.Error("s.topAndHots(%+v) error(%v)", params, err)
  188. err = nil // degrade
  189. }
  190. roots, seconds, total, err := s.rootReplies(c, sub, params.Mid, params.Sort, params.PageNum, params.PageSize, 1, s.sndDefCnt)
  191. if err != nil {
  192. return
  193. }
  194. rs := make([]*model.Reply, 0, len(roots)+len(hots)+len(hseconds)+len(seconds)+2)
  195. rs = append(rs, hots...)
  196. rs = append(rs, roots...)
  197. rs = append(rs, hseconds...)
  198. rs = append(rs, seconds...)
  199. if topAdmin != nil {
  200. rs = append(rs, topAdmin)
  201. }
  202. if topUpper != nil {
  203. rs = append(rs, topUpper)
  204. }
  205. if err = s.buildReply(c, sub, rs, params.Mid, params.Escape); err != nil {
  206. return
  207. }
  208. page = &model.PageResult{
  209. Subject: sub,
  210. TopAdmin: topAdmin,
  211. TopUpper: topUpper,
  212. Hots: hots,
  213. Roots: roots,
  214. Total: total,
  215. AllCount: sub.ACount,
  216. }
  217. return
  218. }
  219. func (s *Service) topAndHots(c context.Context, sub *model.Subject, mid int64, needNot, needSnd bool) (topAdmin, topUpper *model.Reply, hots, seconds []*model.Reply, err error) {
  220. var (
  221. ok bool
  222. hotIDs []int64
  223. rootMap map[int64]*model.Reply
  224. secondMap map[int64][]*model.Reply
  225. )
  226. // get hot replies
  227. if needNot {
  228. if hotIDs, _, err = s.rootReplyIDs(c, sub, model.SortByLike, 1, s.hotNumWeb(sub.Oid, sub.Type)+2, false); err != nil {
  229. return
  230. }
  231. if rootMap, err = s.repliesMap(c, sub.Oid, sub.Type, hotIDs); err != nil {
  232. return
  233. }
  234. }
  235. // get top replies
  236. if topAdmin, err = s.topReply(c, sub, model.SubAttrAdminTop); err != nil {
  237. return
  238. }
  239. if topUpper, err = s.topReply(c, sub, model.SubAttrUpperTop); err != nil {
  240. return
  241. }
  242. if rootMap == nil {
  243. rootMap = make(map[int64]*model.Reply)
  244. }
  245. if topAdmin != nil {
  246. rootMap[topAdmin.RpID] = topAdmin
  247. }
  248. if topUpper != nil {
  249. if !topUpper.IsNormal() && sub.Mid != mid {
  250. topUpper = nil
  251. } else {
  252. rootMap[topUpper.RpID] = topUpper
  253. }
  254. }
  255. // get second replies
  256. if needSnd {
  257. if secondMap, seconds, err = s.secondReplies(c, sub, rootMap, mid, 1, s.sndDefCnt); err != nil {
  258. return
  259. }
  260. if topAdmin != nil {
  261. if topAdmin.Replies, ok = secondMap[topAdmin.RpID]; !ok {
  262. topAdmin.Replies = _emptyReplies
  263. }
  264. }
  265. if topUpper != nil {
  266. if topUpper.Replies, ok = secondMap[topUpper.RpID]; !ok {
  267. topUpper.Replies = _emptyReplies
  268. }
  269. }
  270. }
  271. if len(hotIDs) == 0 {
  272. hots = _emptyReplies
  273. return
  274. }
  275. hotSize := s.hotNumWeb(sub.Oid, sub.Type)
  276. for _, rootID := range hotIDs {
  277. if hotSize != _hotSizeWeb && len(hots) >= _hotSizeWeb {
  278. break
  279. } else if len(hots) >= hotSize {
  280. break
  281. }
  282. if rp, ok := rootMap[rootID]; ok && rp.Like >= _hotLikes && !rp.IsTop() {
  283. if rp.Replies, ok = secondMap[rp.RpID]; !ok {
  284. rp.Replies = _emptyReplies
  285. }
  286. hots = append(hots, rp)
  287. }
  288. }
  289. return
  290. }
  291. func (s *Service) rootReplies(c context.Context, sub *model.Subject, mid int64, msort int8, pn, ps, secondPn, secondPs int) (roots, seconds []*model.Reply, total int, err error) {
  292. var (
  293. rootMap map[int64]*model.Reply
  294. )
  295. // get root replies
  296. rootIDs, total, err := s.rootReplyIDs(c, sub, msort, pn, ps, true)
  297. if err != nil {
  298. return
  299. }
  300. if len(rootIDs) > 0 {
  301. if rootMap, err = s.repliesMap(c, sub.Oid, sub.Type, rootIDs); err != nil {
  302. return
  303. }
  304. }
  305. // get pending audit replies
  306. if msort == model.SortByFloor && sub.AttrVal(model.SubAttrAudit) == model.AttrYes {
  307. var (
  308. pendingTotal int
  309. pendingIDs []int64
  310. rootPendingMap map[int64]*model.Reply
  311. )
  312. if rootPendingMap, _, pendingTotal, err = s.userAuditReplies(c, mid, sub.Oid, sub.Type); err != nil {
  313. err = nil // degrade
  314. }
  315. if rootMap == nil {
  316. rootMap = make(map[int64]*model.Reply)
  317. }
  318. for _, rp := range rootPendingMap {
  319. if withinFloor(rootIDs, rp.RpID, pn, ps, false) {
  320. rootMap[rp.RpID] = rp
  321. pendingIDs = append(pendingIDs, rp.RpID)
  322. }
  323. }
  324. if len(pendingIDs) > 0 {
  325. rootIDs = append(rootIDs, pendingIDs...)
  326. sort.Sort(model.DescFloors(rootIDs))
  327. }
  328. sub.ACount += pendingTotal
  329. }
  330. if len(rootIDs) == 0 {
  331. roots = _emptyReplies
  332. return
  333. }
  334. // get second replies
  335. secondMap, seconds, err := s.secondReplies(c, sub, rootMap, mid, secondPn, secondPs)
  336. if err != nil {
  337. return
  338. }
  339. for _, rootID := range rootIDs {
  340. if rp, ok := rootMap[rootID]; ok {
  341. if rp.Replies, ok = secondMap[rp.RpID]; !ok {
  342. rp.Replies = _emptyReplies
  343. }
  344. if msort != model.SortByFloor {
  345. //if not sort by floor,can't contain the top comment
  346. if rp.IsTop() {
  347. continue
  348. }
  349. }
  350. roots = append(roots, rp)
  351. }
  352. }
  353. if roots == nil {
  354. roots = _emptyReplies
  355. }
  356. return
  357. }
  358. func (s *Service) secondReplies(c context.Context, sub *model.Subject, rootMap map[int64]*model.Reply, mid int64, pn, ps int) (res map[int64][]*model.Reply, rs []*model.Reply, err error) {
  359. var (
  360. rootIDs, secondIDs []int64
  361. secondIdxMap map[int64][]int64
  362. secondMap map[int64]*model.Reply
  363. )
  364. for rootID, info := range rootMap {
  365. if info.RCount > 0 {
  366. rootIDs = append(rootIDs, rootID)
  367. }
  368. }
  369. if len(rootIDs) > 0 {
  370. if secondIdxMap, secondIDs, err = s.getIdsByRoots(c, sub.Oid, rootIDs, sub.Type, pn, ps); err != nil {
  371. return
  372. }
  373. if secondMap, err = s.repliesMap(c, sub.Oid, sub.Type, secondIDs); err != nil {
  374. return
  375. }
  376. }
  377. // get pending audit replies
  378. if sub.AttrVal(model.SubAttrAudit) == model.AttrYes {
  379. var secondPendings map[int64][]*model.Reply
  380. if _, secondPendings, _, err = s.userAuditReplies(c, mid, sub.Oid, sub.Type); err != nil {
  381. err = nil // degrade
  382. }
  383. if secondIdxMap == nil {
  384. secondIdxMap = make(map[int64][]int64)
  385. }
  386. if secondMap == nil {
  387. secondMap = make(map[int64]*model.Reply)
  388. }
  389. for rootID, rs := range secondPendings {
  390. var pendingIDs []int64
  391. if r, ok := rootMap[rootID]; ok {
  392. for _, r := range rs {
  393. if withinFloor(secondIdxMap[rootID], r.RpID, pn, ps, true) {
  394. secondMap[r.RpID] = r
  395. pendingIDs = append(pendingIDs, r.RpID)
  396. }
  397. }
  398. r.RCount += len(rs)
  399. }
  400. if len(pendingIDs) > 0 {
  401. secondIdxMap[rootID] = append(secondIdxMap[rootID], pendingIDs...)
  402. sort.Sort(model.AscFloors(secondIdxMap[rootID]))
  403. }
  404. }
  405. }
  406. res = make(map[int64][]*model.Reply, len(secondIdxMap))
  407. for root, idxs := range secondIdxMap {
  408. seconds := make([]*model.Reply, 0, len(idxs))
  409. for _, rpid := range idxs {
  410. if r, ok := secondMap[rpid]; ok {
  411. seconds = append(seconds, r)
  412. }
  413. }
  414. res[root] = seconds
  415. rs = append(rs, seconds...)
  416. }
  417. return
  418. }
  419. // FilDelReply delete reply which is deleted
  420. func (s *Service) FilDelReply(rps []*model.Reply) (filtedRps []*model.Reply) {
  421. for _, rp := range rps {
  422. if !rp.IsDeleted() {
  423. var childs []*model.Reply
  424. for _, child := range rp.Replies {
  425. if !child.IsDeleted() {
  426. childs = append(childs, child)
  427. }
  428. }
  429. rp.Replies = childs
  430. filtedRps = append(filtedRps, rp)
  431. }
  432. }
  433. return
  434. }
  435. // EmojiReplaceI EmojiReplace international
  436. func (s *Service) EmojiReplaceI(mobiAPP string, build int64, roots ...*model.Reply) {
  437. if mobiAPP == "android_i" && build > 1125000 && build < 2005000 {
  438. for _, root := range roots {
  439. if root != nil {
  440. if root.Content != nil {
  441. if emoCodes := _emojiCode.FindAllString(root.Content.Message, -1); len(emoCodes) > 0 {
  442. root.Content.Message = RepressEmotions(root.Content.Message, emoCodes)
  443. }
  444. }
  445. for _, rp := range root.Replies {
  446. if rp.Content != nil {
  447. if emoCodes := _emojiCode.FindAllString(rp.Content.Message, -1); len(emoCodes) > 0 {
  448. rp.Content.Message = RepressEmotions(rp.Content.Message, emoCodes)
  449. }
  450. }
  451. }
  452. }
  453. }
  454. }
  455. }
  456. // EmojiReplace EmojiReplace
  457. func (s *Service) EmojiReplace(plat int8, build int64, roots ...*model.Reply) {
  458. if (plat == model.PlatIPad || plat == model.PlatPadHd || plat == model.PlatIPhone) && build <= 8170 {
  459. for _, root := range roots {
  460. if root != nil {
  461. for _, rp := range root.Replies {
  462. if rp.Content != nil {
  463. if emoCodes := _emojiCode.FindAllString(rp.Content.Message, -1); len(emoCodes) > 0 {
  464. rp.Content.Message = RepressEmotions(rp.Content.Message, emoCodes)
  465. }
  466. }
  467. }
  468. }
  469. }
  470. }
  471. }
  472. func (s *Service) buildReply(c context.Context, sub *model.Subject, replies []*model.Reply, mid int64, escape bool) (err error) {
  473. var (
  474. ok bool
  475. assistMap map[int64]int
  476. actionMap map[int64]int8
  477. blackedMap map[int64]bool
  478. attetionMap map[int64]*accmdl.RelationReply
  479. rpIDs = make([]int64, 0, len(replies))
  480. mids = make([]int64, 0, len(replies))
  481. uniqMids = make(map[int64]struct{}, len(replies))
  482. fansMap map[int64]*model.FansDetail
  483. accMap map[int64]*accmdl.Card
  484. )
  485. if len(replies) == 0 {
  486. return
  487. }
  488. for _, rp := range replies {
  489. if rp.Content != nil {
  490. for _, mid := range rp.Content.Ats {
  491. uniqMids[mid] = struct{}{}
  492. }
  493. }
  494. uniqMids[rp.Mid] = struct{}{}
  495. rpIDs = append(rpIDs, rp.RpID)
  496. }
  497. for mid := range uniqMids {
  498. mids = append(mids, mid)
  499. }
  500. g := errgroup.WithContext(c)
  501. if mid > 0 {
  502. g.Go(func(c context.Context) error {
  503. actionMap, _ = s.actions(c, mid, sub.Oid, rpIDs)
  504. return nil
  505. })
  506. g.Go(func(c context.Context) error {
  507. attetionMap, _ = s.getAttentions(c, mid, mids)
  508. return nil
  509. })
  510. g.Go(func(c context.Context) error {
  511. blackedMap, _ = s.GetBlacklist(c, mid)
  512. return nil
  513. })
  514. }
  515. g.Go(func(c context.Context) error {
  516. accMap, _ = s.getAccInfo(c, mids)
  517. return nil
  518. })
  519. if !s.IsWhiteAid(sub.Oid, sub.Type) {
  520. g.Go(func(c context.Context) error {
  521. fansMap, _ = s.FetchFans(c, mids, sub.Mid)
  522. return nil
  523. })
  524. g.Go(func(c context.Context) error {
  525. assistMap = s.getAssistList(c, sub.Mid)
  526. return nil
  527. })
  528. }
  529. g.Wait()
  530. // set reply info
  531. for _, r := range replies {
  532. r.FillFolder()
  533. r.FillStr(escape)
  534. if r.Content != nil {
  535. r.Content.FillAts(accMap)
  536. }
  537. r.Action = actionMap[r.RpID]
  538. // member info and degrade
  539. r.Member = new(model.Member)
  540. var card *accmdl.Card
  541. if card, ok = accMap[r.Mid]; ok {
  542. r.Member.Info = new(model.Info)
  543. r.Member.Info.FromCard(card)
  544. } else {
  545. r.Member.Info = new(model.Info)
  546. *r.Member.Info = *s.defMember
  547. r.Member.Info.Mid = strconv.FormatInt(r.Mid, 10)
  548. }
  549. if r.Member.FansDetail, ok = fansMap[r.Mid]; ok {
  550. r.FansGrade = r.Member.FansDetail.Status
  551. }
  552. if _, ok = blackedMap[r.Mid]; ok {
  553. r.State = model.ReplyStateBlacklist
  554. }
  555. if _, ok = assistMap[r.Mid]; ok {
  556. r.Assist = 1
  557. }
  558. if attetion, ok := attetionMap[r.Mid]; ok {
  559. if attetion.Following {
  560. r.Member.Following = 1
  561. }
  562. }
  563. // temporary fix situation: rcount < 0
  564. if r.RCount < 0 {
  565. r.RCount = 0
  566. }
  567. }
  568. return
  569. }
  570. func (s *Service) rootReplyIDs(c context.Context, sub *model.Subject, sort int8, pn, ps int, loadCount bool) (rpIDs []int64, count int, err error) {
  571. var (
  572. ok bool
  573. start = (pn - 1) * ps
  574. end = start + ps - 1
  575. )
  576. if count = sub.RCount; start >= count {
  577. return
  578. }
  579. if sort == model.SortByLike {
  580. mid := metadata.Int64(c, metadata.Mid)
  581. res, err := s.replyHotFeed(c, mid, sub.Oid, int(sub.Type), pn, ps)
  582. if err == nil && res.RpIDs != nil && len(res.RpIDs) > 0 {
  583. log.Info("reply-feed(test): reply abtest mid(%d) oid(%d) type (%d) test name(%s) rpIDs(%v)", mid, sub.Oid, sub.Type, res.Name, res.RpIDs)
  584. rpIDs = res.RpIDs
  585. if loadCount {
  586. count = int(res.Count)
  587. }
  588. return rpIDs, count, nil
  589. }
  590. if err != nil {
  591. log.Error("reply-feed error(%v)", err)
  592. err = nil
  593. } else {
  594. log.Info("reply-feed(origin): reply abtest mid(%d) oid(%d) type (%d) test name(%s) rpIDs(%v)", mid, sub.Oid, sub.Type, res.Name, res.RpIDs)
  595. }
  596. }
  597. // expire the index cache
  598. if ok, err = s.dao.Redis.ExpireIndex(c, sub.Oid, sub.Type, sort); err != nil {
  599. log.Error("s.dao.Redis.ExpireIndex(%d,%d,%d) error(%v)", sub.Oid, sub.Type, sort, err)
  600. return
  601. }
  602. if !ok {
  603. // here we can get that Serviceub.RCount > 0
  604. switch sort {
  605. case model.SortByFloor:
  606. s.dao.Databus.RecoverFloorIdx(c, sub.Oid, sub.Type, end+1, false)
  607. rpIDs, err = s.dao.Reply.GetIdsSortFloor(c, sub.Oid, sub.Type, start, ps)
  608. case model.SortByCount:
  609. s.dao.Databus.RecoverIndex(c, sub.Oid, sub.Type, sort)
  610. rpIDs, err = s.dao.Reply.GetIdsSortCount(c, sub.Oid, sub.Type, start, ps)
  611. case model.SortByLike:
  612. s.dao.Databus.RecoverIndex(c, sub.Oid, sub.Type, sort)
  613. if rpIDs, err = s.dao.Reply.GetIdsSortLike(c, sub.Oid, sub.Type, start, ps); err != nil {
  614. return
  615. }
  616. if loadCount {
  617. count, err = s.dao.Reply.CountLike(c, sub.Oid, sub.Type)
  618. }
  619. }
  620. if err != nil {
  621. log.Error("s.rootIDs(%d,%d,%d,%d,%d) error(%v)", sub.Oid, sub.Type, sort, start, ps, err)
  622. return
  623. }
  624. } else {
  625. var isEnd bool
  626. if rpIDs, isEnd, err = s.dao.Redis.Range(c, sub.Oid, sub.Type, sort, start, end); err != nil {
  627. log.Error("s.dao.Redis.Range(%d,%d,%d,%d,%d) error(%v)", sub.Oid, sub.Type, sort, start, end, err)
  628. return
  629. }
  630. if (sort == model.SortByLike || sort == model.SortByCount) && loadCount {
  631. if count, err = s.dao.Redis.CountReplies(c, sub.Oid, sub.Type, sort); err != nil {
  632. log.Error("s.dao.Redis.CountLike(%d,%d,%d) error(%v)", sub.Oid, sub.Type, sort, err)
  633. }
  634. }
  635. if sort == model.SortByFloor && len(rpIDs) < ps && !isEnd {
  636. //The addition and deletion of comments may result in the display of duplicate entries
  637. rpIDs, err = s.dao.Reply.GetIdsSortFloor(c, sub.Oid, sub.Type, start, ps)
  638. if err != nil {
  639. log.Error("s.rootIDs(%d,%d,%d,%d,%d) error(%v)", sub.Oid, sub.Type, sort, start, ps, err)
  640. return
  641. }
  642. s.dao.Databus.RecoverFloorIdx(c, sub.Oid, sub.Type, end+1, false)
  643. }
  644. }
  645. return
  646. }
  647. // topReply return top replies from cache.
  648. func (s *Service) topReply(c context.Context, sub *model.Subject, top uint32) (rp *model.Reply, err error) {
  649. if top != model.ReplyAttrUpperTop && top != model.ReplyAttrAdminTop {
  650. return
  651. }
  652. if sub.AttrVal(top) == model.AttrYes && sub.Meta != "" {
  653. var meta model.SubjectMeta
  654. err = json.Unmarshal([]byte(sub.Meta), &meta)
  655. if err != nil {
  656. log.Error("s.topReply(%d,%d,%d) unmarshal error(%v)", sub.Oid, sub.Type, top, err)
  657. return
  658. }
  659. var rpid int64
  660. if top == model.SubAttrAdminTop && meta.AdminTop != 0 {
  661. rpid = meta.AdminTop
  662. } else if top == model.SubAttrUpperTop && meta.UpperTop != 0 {
  663. rpid = meta.UpperTop
  664. }
  665. if rpid != 0 {
  666. rp, err = s.ReplyContent(c, sub.Oid, rpid, sub.Type)
  667. if err != nil {
  668. log.Error("s.GetReply(%d,%d,%d) error(%v)", sub.Oid, sub.Type, rpid, err)
  669. return
  670. }
  671. if rp == nil {
  672. log.Error("s.GetReply(%d,%d,%d) is nil", sub.Oid, sub.Type, rpid)
  673. }
  674. return
  675. }
  676. }
  677. if sub.AttrVal(top) == model.AttrYes {
  678. if rp, err = s.dao.Mc.GetTop(c, sub.Oid, sub.Type, top); err != nil {
  679. log.Error("s.dao.Mc.GetTop(%d,%d,%d) error(%v)", sub.Oid, sub.Type, top, err)
  680. return
  681. }
  682. if rp == nil {
  683. s.dao.Databus.AddTop(c, sub.Oid, sub.Type, top)
  684. }
  685. }
  686. return
  687. }
  688. func (s *Service) userAuditReplies(c context.Context, mid, oid int64, tp int8) (rootMap map[int64]*model.Reply, secondMap map[int64][]*model.Reply, total int, err error) {
  689. rpIDs, err := s.dao.Redis.UserAuditReplies(c, mid, oid, tp)
  690. if err != nil {
  691. log.Error("s.dao.Redis.Range(%d,%d,%d) error(%v)", oid, tp, mid, err)
  692. return
  693. }
  694. rpMap, err := s.repliesMap(c, oid, tp, rpIDs)
  695. if err != nil {
  696. return
  697. }
  698. total = len(rpMap)
  699. rootMap = make(map[int64]*model.Reply)
  700. secondMap = make(map[int64][]*model.Reply)
  701. for _, rp := range rpMap {
  702. if rp.Root == 0 {
  703. if !rp.IsTop() {
  704. rootMap[rp.RpID] = rp
  705. }
  706. } else {
  707. secondMap[rp.Root] = append(secondMap[rp.Root], rp)
  708. }
  709. }
  710. return
  711. }
  712. // repliesMap multi get reply from cache or db when missed and fill content.
  713. func (s *Service) repliesMap(c context.Context, oid int64, tp int8, rpIDs []int64) (res map[int64]*model.Reply, err error) {
  714. if len(rpIDs) == 0 {
  715. return
  716. }
  717. res, missIDs, err := s.dao.Mc.GetMultiReply(c, rpIDs)
  718. if err != nil {
  719. log.Error("s.dao.Mc.GetMultiReply(%d,%d,%d) error(%v)", oid, tp, rpIDs, err)
  720. err = nil
  721. res = make(map[int64]*model.Reply, len(rpIDs))
  722. missIDs = rpIDs
  723. }
  724. if len(missIDs) > 0 {
  725. var (
  726. mrp map[int64]*model.Reply
  727. mrc map[int64]*model.Content
  728. )
  729. if mrp, err = s.dao.Reply.GetByIds(c, oid, tp, missIDs); err != nil {
  730. log.Error("s.reply.GetByIds(%d,%d,%d) error(%v)", oid, tp, rpIDs, err)
  731. return
  732. }
  733. if mrc, err = s.dao.Content.GetByIds(c, oid, missIDs); err != nil {
  734. log.Error("s.content.GetByIds(%d,%d) error(%v)", oid, rpIDs, err)
  735. return
  736. }
  737. rs := make([]*model.Reply, 0, len(missIDs))
  738. for _, rpID := range missIDs {
  739. if rp, ok := mrp[rpID]; ok {
  740. rp.Content = mrc[rpID]
  741. res[rpID] = rp
  742. rs = append(rs, rp.Clone())
  743. }
  744. }
  745. // asynchronized add reply cache
  746. select {
  747. case s.replyChan <- replyChan{rps: rs}:
  748. default:
  749. log.Warn("s.replyChan is full")
  750. }
  751. }
  752. return
  753. }
  754. // ReplyContent get reply and content.
  755. func (s *Service) ReplyContent(c context.Context, oid, rpID int64, tp int8) (r *model.Reply, err error) {
  756. if r, err = s.dao.Mc.GetReply(c, rpID); err != nil {
  757. log.Error("replyCacheDao.GetReply(%d, %d, %d) error(%v)", oid, rpID, tp, err)
  758. }
  759. if r == nil {
  760. if r, err = s.dao.Reply.Get(c, oid, rpID); err != nil {
  761. log.Error("s.reply.GetReply(%d, %d) error(%v)", oid, rpID, err)
  762. return nil, err
  763. }
  764. if r == nil {
  765. return nil, ecode.ReplyNotExist
  766. }
  767. if r.Content, err = s.dao.Content.Get(c, oid, rpID); err != nil {
  768. return nil, err
  769. }
  770. if err = s.dao.Mc.AddReply(c, r); err != nil {
  771. log.Error("mc.AddReply(%d,%d,%d) error(%v)", oid, rpID, tp, err)
  772. err = nil
  773. }
  774. }
  775. if r.Oid != oid || r.Type != tp {
  776. log.Warn("reply dismatches with parameter, oid: %d, rpID: %d, tp: %d, actual: %d, %d, %d", oid, rpID, tp, r.Oid, r.RpID, r.Type)
  777. return nil, ecode.RequestErr
  778. }
  779. return r, nil
  780. }
  781. func (s *Service) repliesByRoot(c context.Context, oid, root int64, tp int8, pn, ps int) (res []*model.Reply, err error) {
  782. var (
  783. cache bool
  784. rpIDs []int64
  785. start = (pn - 1) * ps
  786. end = start + ps - 1
  787. )
  788. if cache, err = s.dao.Redis.ExpireIndexByRoot(c, root); err != nil {
  789. return
  790. }
  791. if cache {
  792. if rpIDs, err = s.dao.Redis.RangeByRoot(c, root, start, end); err != nil {
  793. log.Error("s.dao.Redis.RangeByRoots() err(%v)", err)
  794. return
  795. }
  796. } else {
  797. if rpIDs, err = s.dao.Reply.GetIdsByRoot(c, oid, root, tp, start, ps); err != nil {
  798. log.Error("s.dao.Reply.GetIdsByRoot(oid %d,tp %d,root %d) err(%v)", oid, tp, root, err)
  799. }
  800. s.dao.Databus.RecoverIndexByRoot(c, oid, root, tp)
  801. }
  802. rs, err := s.repliesMap(c, oid, tp, rpIDs)
  803. if err != nil {
  804. return
  805. }
  806. for _, rpID := range rpIDs {
  807. if rp, ok := rs[rpID]; ok {
  808. res = append(res, rp)
  809. }
  810. }
  811. return
  812. }
  813. // ReplyHots return the hot replies.
  814. func (s *Service) ReplyHots(c context.Context, oid int64, typ int8, pn, ps int) (sub *model.Subject, res []*model.Reply, err error) {
  815. if !model.LegalSubjectType(typ) {
  816. err = ecode.ReplyIllegalSubType
  817. return
  818. }
  819. if sub, err = s.Subject(c, oid, typ); err != nil {
  820. log.Error("s.Subject(%d,%d) error(%v)", oid, typ, err)
  821. return
  822. }
  823. hotIDs, _, err := s.rootReplyIDs(c, sub, model.SortByLike, pn, ps, false)
  824. if err != nil {
  825. return
  826. }
  827. rootMap, err := s.repliesMap(c, sub.Oid, sub.Type, hotIDs)
  828. if err != nil {
  829. return
  830. }
  831. for _, rpID := range hotIDs {
  832. if rp, ok := rootMap[rpID]; ok {
  833. res = append(res, rp)
  834. }
  835. }
  836. if len(res) == 0 {
  837. res = _emptyReplies
  838. }
  839. return
  840. }
  841. // Dialog ...
  842. func (s *Service) Dialog(c context.Context, mid, oid int64, tp int8, root, dialog int64, pn, ps int, escape bool) (rps []*model.Reply, err error) {
  843. var (
  844. start = (pn - 1) * ps
  845. end = start + ps - 1
  846. ok bool
  847. rpIDs []int64
  848. )
  849. if ok, err = s.dao.Redis.ExpireDialogIndex(c, dialog); err != nil {
  850. log.Error("s.dao.Redis.ExpireDialogIndex error (%v)", err)
  851. return
  852. }
  853. if ok {
  854. rpIDs, err = s.dao.Redis.RangeRpsByDialog(c, dialog, start, end)
  855. } else {
  856. s.dao.Databus.RecoverDialogIdx(c, oid, tp, root, dialog)
  857. rpIDs, err = s.dao.Reply.GetIDsByDialog(c, oid, tp, root, dialog, start, ps)
  858. }
  859. if err != nil {
  860. log.Error("range replies by dialog from redis or db error (%v)", err)
  861. return
  862. }
  863. rpMap, err := s.repliesMap(c, oid, tp, rpIDs)
  864. if err != nil {
  865. return
  866. }
  867. for _, rpID := range rpIDs {
  868. if r, ok := rpMap[rpID]; ok {
  869. rps = append(rps, r)
  870. }
  871. }
  872. sub, err := s.Subject(c, oid, tp)
  873. if err != nil {
  874. return
  875. }
  876. for _, rp := range rps {
  877. rp.DialogStr = strconv.FormatInt(rp.Dialog, 10)
  878. }
  879. if err = s.buildReply(c, sub, rps, mid, escape); err != nil {
  880. return
  881. }
  882. return
  883. }