dm.go 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. package model
  2. import (
  3. "fmt"
  4. "hash/crc32"
  5. "strconv"
  6. "go-common/library/time"
  7. "go-common/library/xstr"
  8. )
  9. var (
  10. bAmp = []byte(`&`)
  11. bGt = []byte(`>`)
  12. bLt = []byte(`<`)
  13. bSp = []byte(` `)
  14. // LimitPerMin 每个rank权限允许的发送速度
  15. LimitPerMin = map[int32]int64{
  16. 0: 10,
  17. 10000: 30,
  18. 15000: 30,
  19. 20000: 300,
  20. 25000: 60,
  21. 30000: 300,
  22. 32000: 300,
  23. }
  24. )
  25. // All const variable used in dm2
  26. const (
  27. // <d p="播放时间,弹幕模式,字体大小,颜色,发送时间,弹幕池,用户hash,弹幕id">弹幕内容</d>
  28. _xmlFmt = `<d p="%.5f,%d,%d,%d,%d,%d,%s,%d">%s</d>`
  29. // <d p="播放时间,弹幕模式,字体大小,颜色,发送时间,弹幕池,用户hash,弹幕id,用户id">弹幕内容</d>
  30. _rnameFmt = `<d p="%.5f,%d,%d,%d,%d,%d,%s,%d,%d">%s</d>`
  31. AttrNo = int32(0) // 属性位为0
  32. AttrYes = int32(1) // 属性位为1
  33. AttrProtect = uint(0) // 保护弹幕
  34. StateNormal = int32(0) // 普通状态
  35. StateDelete = int32(1) // 删除状态
  36. StateHide = int32(2) // 隐藏状态
  37. StateBlock = int32(3) // 屏蔽状态
  38. StateFilter = int32(4) // 过滤状态
  39. StateMonitorBefore = int32(5) // 先审后发
  40. StateMonitorAfter = int32(6) // 先发后审
  41. StateSystemFilter = int32(7) // 敏感词过滤
  42. StateReportDelete = int32(8) // 举报删除
  43. StateAdminDelete = int32(9) // 弹幕管理删除
  44. StateUserDelete = int32(10) // 用户删除
  45. StateScriptDelete = int32(11) // 举报脚本删除
  46. StateTaskDel = int32(12) //弹幕任务删除
  47. StateAiDelete = int32(13) // ai删除
  48. PoolNormal = int32(0) // 普通弹幕池
  49. PoolSubtitle = int32(1) // 字幕弹幕池
  50. PoolSpecial = int32(2) // 特殊弹幕池
  51. PlatUnknow = int32(0)
  52. PlatWeb = int32(1)
  53. PlatAndroid = int32(2)
  54. PlatIPhone = int32(3)
  55. PlatWpM = int32(4) // wp mobile
  56. PlatIPad = int32(5)
  57. PlatPadHd = int32(6) // ipad hd
  58. PlatWpPc = int32(7) // win10
  59. MaxLenDefMsg = 100
  60. MaxLen7Msg = 300
  61. // 弹幕模式
  62. ModeRolling = int32(1)
  63. ModeBottom = int32(4)
  64. ModeTop = int32(5)
  65. ModeReverse = int32(6)
  66. ModeSpecial = int32(7)
  67. ModeCode = int32(8)
  68. ModeBAS = int32(9)
  69. SpamBlack = 52001
  70. SpamOverflow = 52002
  71. SpamRestrict = 52005
  72. )
  73. // AdvanceCmt struct
  74. type AdvanceCmt struct {
  75. ID int64
  76. Owner int64
  77. Oid int64
  78. Type string // request:申请中,buy:已购买,accept:已通过,deny:拒绝
  79. Mode string // 枚举类型,sp:高级弹幕,advance:pool2特殊弹幕
  80. Timestamp int64
  81. Mid int64
  82. Refund int
  83. }
  84. // DailyLimiter daily limiter
  85. type DailyLimiter struct {
  86. Date string
  87. Count int64
  88. }
  89. // Limiter retrun
  90. type Limiter struct {
  91. Allowance int64
  92. Timestamp int64
  93. }
  94. // AttrVal return val of index'attr
  95. func (d *DM) AttrVal(bit uint) int32 {
  96. return (d.Attr >> bit) & int32(1)
  97. }
  98. // AttrSet set val of index'attr
  99. func (d *DM) AttrSet(v int32, bit uint) {
  100. d.Attr = d.Attr&(^(1 << bit)) | (v << bit)
  101. }
  102. // DMSlice dm slice
  103. type DMSlice []*DM
  104. func (d DMSlice) Len() int { return len(d) }
  105. func (d DMSlice) Swap(i, j int) { d[i], d[j] = d[j], d[i] }
  106. func (d DMSlice) Less(i, j int) bool { return d[i].ID < d[j].ID }
  107. // DMSlice2 sort dm slice by progress
  108. type DMSlice2 []*DM
  109. func (d DMSlice2) Len() int { return len(d) }
  110. func (d DMSlice2) Swap(i, j int) { d[i], d[j] = d[j], d[i] }
  111. func (d DMSlice2) Less(i, j int) bool { return d[i].Progress < d[j].Progress }
  112. // JudgeDMList dm list of judge
  113. type JudgeDMList struct {
  114. List []*JDM `json:"list"`
  115. Index []int64 `json:"index"`
  116. }
  117. // JDM judge dm
  118. type JDM struct {
  119. ID int64 `json:"id"`
  120. Msg string `json:"msg"`
  121. Mid int64 `json:"mid"`
  122. Progress string `json:"progress"`
  123. CTime time.Time `json:"ctime"`
  124. }
  125. // FilterData filter-service data
  126. type FilterData struct {
  127. Level int64 `json:"level"`
  128. Limit int64 `json:"limit"`
  129. Msg string `json:"msg"`
  130. TypeID []int64 `json:"typeid"`
  131. Hit []string `json:"hit"`
  132. }
  133. // ToXMLSpecialSeg .
  134. func (d *DM) ToXMLSpecialSeg(realname bool) (s string) {
  135. if d.Content == nil || d.Pool != PoolSpecial {
  136. return
  137. }
  138. msg := d.Content.Msg
  139. if d.ContentSpe != nil {
  140. msg = d.ContentSpe.Msg
  141. }
  142. if len(msg) == 0 {
  143. return
  144. }
  145. s = fmt.Sprintf(`<d id="%d">%s</d>`, d.ID, xmlReplace([]byte(msg)))
  146. return
  147. }
  148. // ToXMLSeg convert dm struct to xml.
  149. func (d *DM) ToXMLSeg(realname bool) (s string) {
  150. if d.Content == nil {
  151. return
  152. }
  153. msg := d.Content.Msg
  154. if d.ContentSpe != nil {
  155. msg = d.ContentSpe.Msg
  156. }
  157. if len(msg) == 0 {
  158. return
  159. }
  160. if d.Pool == PoolSpecial {
  161. msg = ""
  162. }
  163. if realname {
  164. // <d p="弹幕ID,弹幕属性,播放时间,弹幕模式,字体大小,颜色,发送时间,弹幕池,用户hash id,用户mid">弹幕内容</d>
  165. s = fmt.Sprintf(_xmlSegRealnameFmt, d.ID, d.Attr, d.Progress, d.Content.Mode, d.Content.FontSize, d.Content.Color, d.Ctime, d.Pool, Hash(d.Mid, uint32(d.Content.IP)), d.Mid, xmlReplace([]byte(msg)))
  166. } else {
  167. // <d p="弹幕ID,弹幕属性,播放时间,弹幕模式,字体大小,颜色,发送时间,弹幕池,用户id">弹幕内容</d>
  168. s = fmt.Sprintf(_xmlSegFmt, d.ID, d.Attr, d.Progress, d.Content.Mode, d.Content.FontSize, d.Content.Color, d.Ctime, d.Pool, Hash(d.Mid, uint32(d.Content.IP)), xmlReplace([]byte(msg)))
  169. }
  170. return
  171. }
  172. // ToXML convert dm struct to xml.
  173. func (d *DM) ToXML(realname bool) (s string) {
  174. if d.Content == nil {
  175. return
  176. }
  177. msg := d.Content.Msg
  178. if d.ContentSpe != nil {
  179. msg = d.ContentSpe.Msg
  180. }
  181. if len(msg) == 0 {
  182. return
  183. }
  184. if realname {
  185. // <d p="播放时间,弹幕模式,字体大小,颜色,发送时间,弹幕池,用户hash,弹幕id,用户id">弹幕内容</d>
  186. s = fmt.Sprintf(_rnameFmt, float64(d.Progress)/1000.0, d.Content.Mode, d.Content.FontSize, d.Content.Color, d.Ctime, d.Pool, Hash(d.Mid, uint32(d.Content.IP)), d.ID, d.Mid, xmlReplace([]byte(msg)))
  187. } else {
  188. // <d p="播放时间,弹幕模式,字体大小,颜色,发送时间,弹幕池,用户hash,弹幕id">弹幕内容</d>
  189. s = fmt.Sprintf(_xmlFmt, float64(d.Progress)/1000.0, d.Content.Mode, d.Content.FontSize, d.Content.Color, d.Ctime, d.Pool, Hash(d.Mid, uint32(d.Content.IP)), d.ID, xmlReplace([]byte(msg)))
  190. }
  191. return
  192. }
  193. // xmlReplace replace special char in xml.
  194. func xmlReplace(bi []byte) (bo []byte) {
  195. for _, b := range bi {
  196. if b == 0 {
  197. continue
  198. } else if b == '&' {
  199. bo = append(bo, bAmp...)
  200. continue
  201. } else if b == '>' {
  202. bo = append(bo, bGt...)
  203. continue
  204. } else if b == '<' {
  205. bo = append(bo, bLt...)
  206. continue
  207. } else if (b >= 0x01 && b <= 0x08) || (b >= 0x0b && b <= 0x0c) || (b >= 0x0e && b <= 0x1f) || (b == 0x7f) {
  208. // 替换掉控制字符,保留空格、回车、制表符
  209. bo = append(bo, bSp...)
  210. } else {
  211. bo = append(bo, b)
  212. }
  213. }
  214. return
  215. }
  216. // Hash return mid hash string.
  217. func Hash(mid int64, ip uint32) string {
  218. var s uint32
  219. if mid != 0 {
  220. s = crc32.ChecksumIEEE([]byte(strconv.FormatInt(mid, 10)))
  221. return strconv.FormatInt(int64(s), 16)
  222. }
  223. s = crc32.ChecksumIEEE([]byte(strconv.FormatInt(int64(ip), 10)))
  224. return "D" + strconv.FormatInt(int64(s), 16)
  225. }
  226. // IsDMVisible dm can visible
  227. func IsDMVisible(state int32) bool {
  228. if state == StateNormal || state == StateMonitorAfter {
  229. return true
  230. }
  231. return false
  232. }
  233. // IsDMEditAble dm can edit
  234. func IsDMEditAble(state int32) bool {
  235. if state == StateNormal || state == StateHide || state == StateMonitorAfter {
  236. return true
  237. }
  238. return false
  239. }
  240. // AttrNtoA convert uint to string format,eg:5-->101-->1,3.
  241. func (d *DM) AttrNtoA() string {
  242. if d.Attr == 0 {
  243. return ""
  244. }
  245. var bits []int64
  246. for k, v := range strconv.FormatInt(int64(d.Attr), 2) {
  247. if v == 49 {
  248. bits = append(bits, int64(k+1))
  249. }
  250. }
  251. return xstr.JoinInts(bits)
  252. }
  253. // DMAttrNtoA convert uint to string format,eg:5-->101-->1,3.
  254. func DMAttrNtoA(attr int32) string {
  255. if attr == 0 {
  256. return ""
  257. }
  258. var bits []int64
  259. for k, v := range strconv.FormatInt(int64(attr), 2) {
  260. if v == 49 {
  261. bits = append(bits, int64(k+1))
  262. }
  263. }
  264. return xstr.JoinInts(bits)
  265. }