dm_history.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. package service
  2. import (
  3. "bytes"
  4. "context"
  5. "fmt"
  6. "math"
  7. "go-common/app/interface/main/dm2/model"
  8. "go-common/library/log"
  9. )
  10. const (
  11. _hisPagesize = 5000
  12. )
  13. // SearchDMHisIndex get history date index.
  14. func (s *Service) SearchDMHisIndex(c context.Context, tp int32, oid int64, month string) (dates []string, err error) {
  15. if dates, err = s.dao.HistoryIdxCache(c, tp, oid, month); err == nil && len(dates) > 0 {
  16. return
  17. }
  18. if dates, err = s.dao.SearchDMHisIndex(c, tp, oid, month); err != nil {
  19. log.Error("dao.SearchDMHisIndex(%d,%d,%s) error(%v)", tp, oid, month, err)
  20. return
  21. }
  22. if len(dates) > 0 {
  23. s.cache.Do(c, func(ctx context.Context) {
  24. s.dao.AddHisIdxCache(ctx, tp, oid, month, dates)
  25. })
  26. }
  27. return
  28. }
  29. // SearchDMHistory get history dm list from search.
  30. func (s *Service) SearchDMHistory(c context.Context, tp int32, oid, ctimeTo int64) (xml []byte, err error) {
  31. var (
  32. sub *model.Subject
  33. dmids []int64
  34. contentSpeMap = make(map[int64]*model.ContentSpecial)
  35. )
  36. if xml, err = s.dao.HistoryCache(c, tp, oid, ctimeTo); err == nil && len(xml) > 0 {
  37. return
  38. }
  39. if sub, err = s.subject(c, tp, oid); err != nil {
  40. return
  41. }
  42. buf := new(bytes.Buffer)
  43. defer func() {
  44. if err == nil {
  45. buf.WriteString(`</i>`)
  46. xml, err = s.gzflate(buf.Bytes())
  47. s.cache.Do(c, func(ctx context.Context) {
  48. s.dao.AddHistoryCache(ctx, tp, oid, ctimeTo, xml)
  49. })
  50. }
  51. }()
  52. buf.WriteString(`<?xml version="1.0" encoding="UTF-8"?><i>`)
  53. buf.WriteString(`<chatserver>chat.bilibili.com</chatserver><chatid>`)
  54. buf.WriteString(fmt.Sprint(sub.Oid))
  55. buf.WriteString(`</chatid><mission>`)
  56. buf.WriteString(fmt.Sprint(sub.AttrVal(model.AttrSubMission)))
  57. buf.WriteString(`</mission><maxlimit>`)
  58. buf.WriteString(fmt.Sprint(sub.Maxlimit))
  59. buf.WriteString(`</maxlimit>`)
  60. buf.WriteString(fmt.Sprintf(`<state>%d</state>`, sub.State))
  61. realname := s.isRealname(c, sub.Pid, sub.Oid)
  62. if realname {
  63. buf.WriteString(`<real_name>1</real_name>`)
  64. } else {
  65. buf.WriteString(`<real_name>0</real_name>`)
  66. }
  67. if sub.State == model.SubStateClosed {
  68. return
  69. }
  70. totalPage := int(math.Ceil(float64(sub.Maxlimit) / float64(_hisPagesize)))
  71. dmids, err = s.dao.SearchDMHistory(c, tp, oid, ctimeTo, totalPage, _hisPagesize)
  72. if err != nil {
  73. return
  74. }
  75. if len(dmids) == 0 {
  76. return
  77. }
  78. if int64(len(dmids)) > sub.Maxlimit {
  79. dmids = dmids[:sub.Maxlimit]
  80. }
  81. idxMap, special, err := s.dao.IndexsByid(c, tp, oid, dmids)
  82. if err != nil {
  83. return
  84. }
  85. ctsMap, err := s.dao.Contents(c, oid, dmids)
  86. if err != nil {
  87. return
  88. }
  89. if len(special) > 0 {
  90. if contentSpeMap, err = s.dao.ContentsSpecial(c, special); err != nil {
  91. return
  92. }
  93. }
  94. for _, dmid := range dmids {
  95. if idx, ok := idxMap[dmid]; ok {
  96. dm := &model.DM{
  97. ID: idx.ID,
  98. Type: idx.Type,
  99. Oid: idx.Oid,
  100. Mid: idx.Mid,
  101. Progress: idx.Progress,
  102. Pool: idx.Pool,
  103. Attr: idx.Attr,
  104. State: idx.State,
  105. Ctime: idx.Ctime,
  106. Mtime: idx.Mtime,
  107. }
  108. content, ok := ctsMap[dmid]
  109. if !ok {
  110. continue
  111. }
  112. dm.Content = content
  113. if idx.Pool == model.PoolSpecial {
  114. if ctsSpec, ok := contentSpeMap[dmid]; ok {
  115. dm.ContentSpe = ctsSpec
  116. }
  117. }
  118. buf.WriteString(dm.ToXML(realname))
  119. }
  120. }
  121. return
  122. }
  123. // SearchDMHistoryV2 get history dm list from search.
  124. func (s *Service) SearchDMHistoryV2(c context.Context, tp int32, oid, ctimeTo int64) (res *model.DMSeg, err error) {
  125. sub, err := s.subject(c, tp, oid)
  126. if err != nil {
  127. return
  128. }
  129. res = &model.DMSeg{Elems: make([]*model.Elem, 0, sub.Maxlimit)}
  130. if sub.State == model.SubStateClosed {
  131. return
  132. }
  133. totalPage := int(math.Ceil(float64(sub.Maxlimit) / float64(_hisPagesize)))
  134. dmids, err := s.dao.SearchDMHistory(c, tp, oid, ctimeTo, totalPage, _hisPagesize)
  135. if err != nil {
  136. return
  137. }
  138. if len(dmids) == 0 {
  139. fmt.Println("dmids from search is empty")
  140. return
  141. }
  142. if int64(len(dmids)) > sub.Maxlimit {
  143. dmids = dmids[:sub.Maxlimit]
  144. }
  145. // TODO special dm
  146. idxMap, _, err := s.dao.IndexsByid(c, tp, oid, dmids)
  147. if err != nil {
  148. return
  149. }
  150. ctsMap, err := s.dao.Contents(c, oid, dmids)
  151. if err != nil {
  152. return
  153. }
  154. for _, dmid := range dmids {
  155. if idx, ok := idxMap[dmid]; ok {
  156. dm := &model.DM{
  157. ID: idx.ID,
  158. Type: idx.Type,
  159. Oid: idx.Oid,
  160. Mid: idx.Mid,
  161. Progress: idx.Progress,
  162. Pool: idx.Pool,
  163. Attr: idx.Attr,
  164. State: idx.State,
  165. Ctime: idx.Ctime,
  166. Mtime: idx.Mtime,
  167. }
  168. content, ok := ctsMap[dmid]
  169. if !ok {
  170. continue
  171. }
  172. dm.Content = content
  173. if e := dm.ToElem(); e != nil {
  174. res.Elems = append(res.Elems, e)
  175. }
  176. }
  177. }
  178. return
  179. }