reply.go 42 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477
  1. package service
  2. import (
  3. "context"
  4. "regexp"
  5. "strconv"
  6. "strings"
  7. "sync"
  8. "time"
  9. "go-common/app/interface/main/reply/conf"
  10. "go-common/app/interface/main/reply/model/adminlog"
  11. "go-common/app/interface/main/reply/model/drawyoo"
  12. "go-common/app/interface/main/reply/model/reply"
  13. accmdl "go-common/app/service/main/account/api"
  14. "go-common/app/service/main/archive/api"
  15. arcmdl "go-common/app/service/main/archive/model/archive"
  16. assmdl "go-common/app/service/main/assist/model/assist"
  17. filgrpc "go-common/app/service/main/filter/api/grpc/v1"
  18. locmdl "go-common/app/service/main/location/model"
  19. relmdl "go-common/app/service/main/relation/model"
  20. thumdl "go-common/app/service/main/thumbup/model"
  21. ugcpay "go-common/app/service/main/ugcpay/api/grpc/v1"
  22. "go-common/library/ecode"
  23. "go-common/library/log"
  24. xip "go-common/library/net/ip"
  25. "go-common/library/net/metadata"
  26. "go-common/library/queue/databus/report"
  27. "go-common/library/sync/errgroup"
  28. xtime "go-common/library/time"
  29. "github.com/mvdan/xurls"
  30. )
  31. var (
  32. _emptyReplies = make([]*reply.Reply, 0)
  33. _emptyAction = map[int64]int8{}
  34. _emptyCards = make(map[int64]*accmdl.Card)
  35. _emptyBlackList = make(map[int64]bool)
  36. _emptyRelations = make(map[int64]*accmdl.RelationReply)
  37. _emojiCode = regexp.MustCompile(`\[[^\[+][^]]+]`)
  38. )
  39. // status
  40. const (
  41. StatusNormal = 1
  42. StatusNeedContest = 2
  43. StatusForbidden = 3
  44. )
  45. // IsWhiteAid IsWhiteAid
  46. func (s *Service) IsWhiteAid(aid int64, tp int8) bool {
  47. if tp != 1 {
  48. return false
  49. }
  50. for _, white := range s.aidWhiteList {
  51. if aid == white {
  52. return true
  53. }
  54. }
  55. return false
  56. }
  57. // UserBlockStatus UserBlockStatus
  58. func (s *Service) UserBlockStatus(c context.Context, mid int64) (int, error) {
  59. res, err := s.dao.BlockStatus.BlockInfo(c, mid)
  60. if err != nil {
  61. return 0, err
  62. }
  63. if res.ForeverBlock || time.Now().Unix() < res.BlockUntil {
  64. return StatusForbidden, nil
  65. }
  66. if res.PassTest == 0 {
  67. return StatusNormal, nil
  68. }
  69. return StatusNeedContest, nil
  70. }
  71. // ValidUserStatus validate reply user status
  72. func (s *Service) ValidUserStatus(c context.Context, profile *accmdl.Profile, isUpper bool) (err error) {
  73. // if myInfo.Silence == 1 {
  74. // err = ecode.UserDisabled
  75. // } else if myInfo.Active == 0 {
  76. // err = ecode.UserInactive
  77. // } else if myInfo.Moral < 60 {
  78. // err = ecode.LackOfScores
  79. // } else if myInfo.Rank == 5000 {
  80. // err = ecode.UserNoMember
  81. // } else if myInfo.Level.Cur < 1 {
  82. // err = ecode.UserLevelLow
  83. // }
  84. if profile.Silence == 1 {
  85. err = ecode.UserDisabled
  86. } else if profile.TelStatus == 0 && profile.EmailStatus == 0 {
  87. err = ecode.UserInactive
  88. } else if profile.Moral < 60 {
  89. err = ecode.LackOfScores
  90. } else if profile.Rank == 5000 && !isUpper {
  91. err = ecode.UserNoMember
  92. } else if profile.Level < 1 && !isUpper {
  93. err = ecode.UserLevelLow
  94. }
  95. return
  96. }
  97. // ValidUserAction validate reply user status for like/hate action.
  98. func (s *Service) ValidUserAction(c context.Context, profile *accmdl.Profile) (err error) {
  99. if profile.Silence == 1 {
  100. err = ecode.UserDisabled
  101. } else if profile.TelStatus == 0 && profile.EmailStatus == 0 {
  102. err = ecode.UserInactive
  103. } else if profile.Moral < 60 {
  104. err = ecode.LackOfScores
  105. }
  106. return
  107. }
  108. // checkSpam detemine whether user can reply or not
  109. func (s *Service) checkSpam(c context.Context, sub *reply.Subject, mid int64, captcha string, level int) (uri string, err error) {
  110. if sub.Type != reply.SubTypeBBQ && sub.Type != reply.SubTypeHuoniao {
  111. if level <= reply.UserLevelFirst && sub.Mid != mid {
  112. if captcha == "" {
  113. var uri string
  114. uri, err = s.Captcha(c, mid)
  115. if err != nil {
  116. return "", err
  117. }
  118. return uri, ecode.ReplyDeniedAsCaptcha
  119. } else if err = s.VerifyCaptcha(c, captcha, mid); err != nil {
  120. return "", err
  121. }
  122. }
  123. }
  124. recent, daily, err := s.dao.Redis.SpamReply(c, mid)
  125. if err != nil {
  126. log.Error("replyCacheDao.SpamReply(%d), err (%v)", mid, err)
  127. return "", err
  128. }
  129. if recent == ecode.ReplyDeniedAsCD.Code() || daily == ecode.ReplyDeniedAsCD.Code() {
  130. return "", ecode.ReplyDeniedAsCD
  131. }
  132. if recent == ecode.ReplyDeniedAsCaptcha.Code() || daily == ecode.ReplyDeniedAsCaptcha.Code() {
  133. if captcha == "" {
  134. uri, err := s.Captcha(c, mid)
  135. if err != nil {
  136. return "", err
  137. }
  138. return uri, ecode.ReplyDeniedAsCaptcha
  139. }
  140. if err := s.VerifyCaptcha(c, captcha, mid); err != nil {
  141. return "", err
  142. }
  143. s.dao.Redis.DelReplyIncr(c, mid, sub.Mid == mid)
  144. s.dao.Redis.DelReplySpam(c, mid)
  145. }
  146. s.dao.Databus.AddSpam(c, sub.Oid, mid, sub.Mid == mid, sub.Type)
  147. return "", nil
  148. }
  149. func (s *Service) isNormalVip(c context.Context, profile *accmdl.Profile) bool {
  150. return profile.Vip.Type != 0 && profile.Vip.Status == 1
  151. }
  152. // ContainUrls ContainUrls
  153. func (s *Service) ContainUrls(msg string) bool {
  154. return xurls.Strict.FindAllString(msg, -1) != nil
  155. }
  156. // bigDataFilter check content by big data and find conmment garbage
  157. func (s *Service) bigDataFilter(c context.Context, msg string) (err error) {
  158. if err = s.bigdata.Filter(c, msg); err != nil {
  159. log.Error("s.bigdata.Filter(%s) error(%v)", msg, err)
  160. }
  161. return
  162. }
  163. func (s *Service) isUpper(c context.Context, mid, oid int64, tp int8) bool {
  164. sub, err := s.getSubject(c, oid, tp)
  165. if err != nil {
  166. return false
  167. }
  168. return sub.Mid == mid
  169. }
  170. // CheckAssist check whether upper grant the supervision permission for user
  171. func (s *Service) CheckAssist(c context.Context, mid, uid int64) (assisted bool, operation bool) {
  172. arg := &assmdl.ArgAssist{
  173. Mid: mid,
  174. AssistMid: uid,
  175. Type: 1,
  176. RealIP: "",
  177. }
  178. if respro, _ := s.assist.Assist(c, arg); respro == nil {
  179. log.Error("s.assist.Assist(%d, %d) error(%v)", mid, uid, "获取up协管关系错误")
  180. } else if respro.Assist == 1 {
  181. assisted = true
  182. if respro.Allow == 1 {
  183. operation = true
  184. }
  185. }
  186. return assisted, operation
  187. }
  188. // getAssistList fetch all assistants of user mid
  189. func (s *Service) getAssistList(c context.Context, mid int64) (assistMap map[int64]int) {
  190. ip := metadata.String(c, metadata.RemoteIP)
  191. arg := &assmdl.ArgAssists{
  192. Mid: mid,
  193. RealIP: ip,
  194. }
  195. assistMap = make(map[int64]int)
  196. if response, err := s.assist.AssistIDs(c, arg); err != nil {
  197. log.Error("s.assist.Assists(%d) error(%v)", mid, err)
  198. } else {
  199. for _, tmp := range response {
  200. assistMap[tmp] = 1
  201. }
  202. }
  203. return
  204. }
  205. // checkContentFilter2 check content by word filter and minus moral when this be filtered.
  206. func (s *Service) checkContentFilter2(c context.Context, oid, mid, rpid int64, ip, msg string, tp int8) (correct string, err error) {
  207. arg := &filgrpc.FilterReq{
  208. Message: msg,
  209. Area: "reply",
  210. Id: rpid,
  211. Oid: oid,
  212. Mid: mid,
  213. }
  214. var res *filgrpc.FilterReply
  215. res, err = s.filcli.Filter(c, arg)
  216. if err != nil {
  217. log.Error("s.filter.Filter(%s) error(%v)", msg, err)
  218. return msg, err
  219. }
  220. switch int(res.Limit) {
  221. case ecode.FilterHitLimitBlack.Code():
  222. log.Info("Reply id %d, content %q contains sensitive msg, not allowed to send out", rpid, msg)
  223. err = ecode.ReplyHitBlacklist
  224. case ecode.FilterHitRubLimit.Code():
  225. log.Info("Reply id %d, content %q was sent too many times, exceed allowed counts", rpid, msg)
  226. err = ecode.ReplyOverRateLimit
  227. case ecode.FilterHitStrictLimit.Code():
  228. if res.Level == 0 {
  229. log.Info("Reply id %d, content %q was limit strictly", rpid, msg)
  230. err = ecode.ReplyDeniedAsCaptcha
  231. }
  232. }
  233. if err != nil {
  234. return msg, err
  235. }
  236. if res.Level > 0 {
  237. s.cache.Do(c, func(ctx context.Context) {
  238. s.AddFilteredReply(ctx, rpid, oid, mid, tp, int8(res.Level), msg, time.Now())
  239. })
  240. switch res.Level {
  241. case 10, 20:
  242. err = ecode.ReplyMosaicByFilter
  243. case 30:
  244. err = ecode.ReplyDeniedByFilter
  245. return
  246. case 40:
  247. tmp := []rune(msg)
  248. if len(tmp) > 80 {
  249. tmp = tmp[:80]
  250. }
  251. arg := &accmdl.MoralReq{
  252. Mid: mid,
  253. Moral: -1,
  254. Oper: "",
  255. Reason: "发布恶意评论: " + string(tmp),
  256. Remark: "云屏蔽",
  257. RealIp: ip,
  258. }
  259. if _, err = s.acc.AddMoral3(c, arg); err != nil {
  260. log.Error("s.acc.AddMoral3(%d) error(%v)", mid, err)
  261. return
  262. }
  263. err = ecode.ReplyDeniedByFilter
  264. return
  265. }
  266. }
  267. correct = res.Result
  268. return
  269. }
  270. // AddFilteredReply AddFilteredReply
  271. func (s *Service) AddFilteredReply(c context.Context, rpID, oid, mid int64, tp, level int8, message string, now time.Time) (err error) {
  272. return s.dao.Reply.AddFilteredReply(c, rpID, oid, mid, tp, level, message, now)
  273. }
  274. // UseBigdata use bigdata switch
  275. func (s *Service) UseBigdata(c context.Context, b bool, per int64) bool {
  276. s.useBigData = b
  277. return s.useBigData
  278. }
  279. // AddReply add a reply.
  280. func (s *Service) AddReply(c context.Context, mid, oid int64, tp, plat int8, ats []int64, accessKey, cookie, captcha, msg, dev, ver, platform string, build int64, buvid string) (r *reply.Reply, uri string, err error) {
  281. var (
  282. rootID, parentID, dialog int64
  283. profile *accmdl.Profile
  284. subject *reply.Subject
  285. )
  286. //whitelist for test
  287. profile, subject, uri, err = s.validateReply(c, mid, oid, tp, captcha, msg, accessKey, cookie)
  288. if err != nil {
  289. return
  290. }
  291. // check content contain emoji code
  292. if emoCodes := _emojiCode.FindAllString(msg, -1); len(emoCodes) > 0 {
  293. if s.isNormalVip(c, profile) {
  294. if len(emoCodes) > conf.Conf.Reply.MaxEmoji {
  295. err = ecode.ReplyEmojiOverMax
  296. return
  297. }
  298. needRepressEmoCodes := make([]string, 0)
  299. for _, emoCode := range emoCodes {
  300. if _, ok := s.emojisM[emoCode]; !ok {
  301. needRepressEmoCodes = append(needRepressEmoCodes, emoCode)
  302. }
  303. }
  304. if len(needRepressEmoCodes) > 0 {
  305. msg = RepressEmotions(msg, needRepressEmoCodes)
  306. }
  307. } else {
  308. msg = RepressEmotions(msg, emoCodes)
  309. }
  310. }
  311. if err = s.SuperviseReply(c, mid, accessKey, cookie, int8(tp)); err != nil {
  312. return
  313. }
  314. r, err = s.persistReply(c, mid, rootID, parentID, plat, tp, ats, msg, dev, ver, captcha, platform, build, buvid, subject, dialog)
  315. if err == ecode.ReplyDeniedAsCaptcha {
  316. uri, err := s.Captcha(c, mid)
  317. if err != nil {
  318. return r, "", err
  319. }
  320. return r, uri, ecode.ReplyDeniedAsCaptcha
  321. }
  322. return
  323. }
  324. // RepressEmotions RepressEmotions
  325. func RepressEmotions(msg string, emoCodes []string) string {
  326. for _, emoCode := range emoCodes {
  327. msg = repressEmotion(msg, emoCode)
  328. }
  329. return msg
  330. }
  331. func repressEmotion(msg, emoCode string) string {
  332. // replace [] to 【】
  333. emoCode = emoCode[1 : len(emoCode)-1]
  334. return strings.Replace(msg, "["+emoCode+"]", "【"+emoCode+"】", -1)
  335. }
  336. // AddReplyReply add reply to a root reply.
  337. func (s *Service) AddReplyReply(c context.Context, mid, oid, rootID, parentID int64, tp, plat int8, ats []int64, accessKey, cookie, captcha, msg, dev, ver, platform string, build int64, buvid string) (r *reply.Reply, uri string, err error) {
  338. var dialog int64
  339. var profile *accmdl.Profile
  340. var subject *reply.Subject
  341. profile, subject, uri, err = s.validateReply(c, mid, oid, tp, captcha, msg, accessKey, cookie)
  342. if err != nil {
  343. return
  344. }
  345. if emoCodes := _emojiCode.FindAllString(msg, -1); len(emoCodes) > 0 {
  346. if s.isNormalVip(c, profile) {
  347. if len(emoCodes) > conf.Conf.Reply.MaxEmoji {
  348. err = ecode.ReplyEmojiOverMax
  349. return
  350. }
  351. needRepressEmoCodes := make([]string, 0)
  352. for _, emoCode := range emoCodes {
  353. if _, ok := s.emojisM[emoCode]; !ok {
  354. needRepressEmoCodes = append(needRepressEmoCodes, emoCode)
  355. }
  356. }
  357. if len(needRepressEmoCodes) > 0 {
  358. msg = RepressEmotions(msg, needRepressEmoCodes)
  359. }
  360. } else {
  361. msg = RepressEmotions(msg, emoCodes)
  362. }
  363. }
  364. rootReply, err := s.GetRootReply(c, oid, rootID, tp)
  365. if err != nil {
  366. return
  367. }
  368. // NOTE if the pending reply, the state is not normal
  369. if rootReply.IsDeleted() {
  370. err = ecode.ReplyDeleted
  371. return
  372. }
  373. if s.RelationBlocked(c, rootReply.Mid, mid) {
  374. err = ecode.ReplyBlacklistFilter
  375. return
  376. }
  377. if err = s.SuperviseReply(c, mid, accessKey, cookie, int8(tp)); err != nil {
  378. return
  379. }
  380. if rootID != parentID {
  381. var parentReply *reply.Reply
  382. if parentReply, err = s.GetReply(c, oid, parentID, tp); err != nil {
  383. return
  384. }
  385. // if parentReply.Dialog == 0 {
  386. // s.dao.Databus.RecoverFixDialogIdx(c, oid, tp, rootID)
  387. // }
  388. dialog = parentReply.Dialog
  389. if parentReply.Root != rootID {
  390. err = ecode.ReplyIllegalRoot
  391. return
  392. }
  393. if mid != parentReply.Mid && !parentReply.IsNormal() {
  394. err = ecode.ReplyNotExist
  395. return
  396. }
  397. if s.RelationBlocked(c, parentReply.Mid, mid) {
  398. err = ecode.ReplyBlacklistFilter
  399. return
  400. }
  401. }
  402. r, err = s.persistReply(c, mid, rootID, parentID, plat, tp, ats, msg, dev, ver, captcha, platform, build, buvid, subject, dialog)
  403. if err == ecode.ReplyDeniedAsCaptcha {
  404. uri, err := s.Captcha(c, mid)
  405. if err != nil {
  406. return r, "", err
  407. }
  408. return r, uri, ecode.ReplyDeniedAsCaptcha
  409. }
  410. return
  411. }
  412. func (s *Service) validateReply(c context.Context, mid, oid int64, tp int8, captcha, msg, accessKey, cookie string) (profile *accmdl.Profile, subject *reply.Subject, uri string, err error) {
  413. profile, err = s.userInfo(c, mid)
  414. if err != nil {
  415. log.Error("myinfo(%d) error(%v)", mid, err)
  416. return nil, nil, "", err
  417. }
  418. if tp != reply.SubTypeBBQ {
  419. if conf.Conf.Identification.SwitchOn && profile.Identification == 0 {
  420. if profile.TelStatus == 0 {
  421. err = ecode.UserCheckNoPhone
  422. return
  423. }
  424. if profile.TelStatus == 2 && profile.Identification == 0 {
  425. err = ecode.UserCheckInvalidPhone
  426. return
  427. }
  428. }
  429. }
  430. subject, err = s.Subject(c, oid, tp)
  431. if err != nil {
  432. return
  433. }
  434. if tp != reply.SubTypeBBQ && tp != reply.SubTypeHuoniao {
  435. if err = s.ValidUserStatus(c, profile, subject.Mid == mid); err != nil {
  436. log.Warn("s.ValidUserStatus(%d,%+v) error(%v)", mid, profile.Level, err)
  437. return
  438. }
  439. }
  440. if s.RelationBlocked(c, subject.Mid, mid) {
  441. err = ecode.ReplyBlacklistFilter
  442. return
  443. }
  444. if tp != reply.SubTypeBBQ && tp != reply.SubTypeHuoniao {
  445. if profile.Level < reply.UserLevelSnd && subject.Mid != mid {
  446. err = ecode.UserLevelLow
  447. return
  448. }
  449. }
  450. if mid != 165252 && mid != 10287644 {
  451. if uri, err = s.checkSpam(c, subject, mid, captcha, int(profile.Level)); err != nil {
  452. log.Error("s.checkSpam failed(%d) err is %V", mid, err)
  453. return
  454. }
  455. }
  456. if tp == reply.SubTypeArchive && subject.Mid != mid {
  457. var arc *api.Arc
  458. arc, err = s.arcSrv.Archive3(c, &arcmdl.ArgAid2{Aid: oid})
  459. if err != nil {
  460. log.Error("s.arcSrc.Archive3(%d) failed!err:=%v", oid, err)
  461. return
  462. }
  463. if arc.Rights.UGCPay == 1 {
  464. var relation *ugcpay.AssetRelationResp
  465. relation, err = s.ugcpay.AssetRelation(c, &ugcpay.AssetRelationReq{Mid: mid, Oid: oid, Otype: "archive"})
  466. if err != nil {
  467. log.Error("s.ugcpay.AssetRelation(%d,%d) failed!err:=%v", mid, oid, err)
  468. return
  469. }
  470. if relation.State != "paid" {
  471. err = ecode.ReplyForbidReplyNotPay
  472. return
  473. }
  474. }
  475. }
  476. return
  477. }
  478. func (s *Service) persistReply(c context.Context, mid, root, parent int64, plat, tp int8, ats []int64, msg, dev, ver, captcha, platform string, build int64, buvid string, subject *reply.Subject, dialog int64) (r *reply.Reply, err error) {
  479. rpID, err := s.nextID(c)
  480. if err != nil {
  481. return
  482. }
  483. // 一级子评论
  484. if root == parent && root != 0 {
  485. dialog = rpID
  486. } else if root != parent {
  487. parentRp, err := s.reply(c, mid, subject.Oid, parent, tp)
  488. if err != nil {
  489. return nil, err
  490. }
  491. dialog = parentRp.Dialog
  492. }
  493. cTime := xtime.Time(time.Now().Unix())
  494. ip := metadata.String(c, metadata.RemoteIP)
  495. port := metadata.String(c, metadata.RemotePort)
  496. r = &reply.Reply{
  497. RpID: rpID,
  498. Oid: subject.Oid,
  499. Type: tp,
  500. Mid: mid,
  501. Root: root,
  502. State: reply.ReplyStateNormal,
  503. Parent: parent,
  504. CTime: cTime,
  505. Dialog: dialog,
  506. Content: &reply.Content{
  507. RpID: rpID,
  508. Message: msg,
  509. Ats: ats,
  510. IP: xip.InetAtoN(ip),
  511. Plat: plat,
  512. Device: dev,
  513. Version: ver,
  514. CTime: cTime,
  515. },
  516. }
  517. if s.useBigData {
  518. if err = s.bigDataFilter(c, msg); err != nil {
  519. if err == ecode.ReplyDeniedAsGarbage {
  520. // TODO: do not use garbage as state
  521. r.State = reply.ReplyStateGarbage
  522. r.AttrSet(reply.AttrYes, reply.ReplyAttrGarbage)
  523. }
  524. }
  525. }
  526. // if not rpID passed, then no data will be recorded
  527. msg, err = s.checkContentFilter2(c, r.Oid, mid, rpID, ip, msg, r.Type)
  528. if err != nil {
  529. if err != ecode.ReplyDeniedAsCaptcha && err != ecode.ReplyMosaicByFilter {
  530. log.Error("s.checkContentFilter2(%d, %d, msg: %s) error(%v)", mid, subject.Oid, msg, err)
  531. return
  532. }
  533. if err == ecode.ReplyHitBlacklist || err == ecode.ReplyOverRateLimit {
  534. return
  535. }
  536. if err == ecode.ReplyDeniedAsCaptcha {
  537. if captcha == "" {
  538. return
  539. }
  540. if err = s.VerifyCaptcha(c, captcha, mid); err != nil {
  541. return
  542. }
  543. } else {
  544. r.Content.Message = msg
  545. r.AttrSet(reply.AttrYes, reply.ReplyAttrFilter)
  546. r.State = reply.ReplyStateFiltered
  547. }
  548. }
  549. // NOTE audit pending most priority
  550. if subject.AttrVal(reply.SubAttrMonitor) == reply.AttrYes {
  551. r.State = reply.ReplyStateMonitor
  552. }
  553. if subject.AttrVal(reply.SubAttrAudit) == reply.AttrYes {
  554. r.State = reply.ReplyStateAudit
  555. }
  556. s.dao.Databus.AddReply(c, subject.Oid, r)
  557. report.User(&report.UserInfo{
  558. Mid: r.Mid,
  559. Platform: platform,
  560. Build: build,
  561. Buvid: buvid,
  562. Business: 41,
  563. Type: int(r.Type),
  564. Oid: r.Oid,
  565. Action: reply.ReportReplyAdd,
  566. Ctime: time.Now(),
  567. IP: ip + ":" + port,
  568. Index: []interface{}{
  569. r.RpID,
  570. r.State,
  571. r.State,
  572. strconv.FormatInt(r.Root, 10),
  573. },
  574. })
  575. return
  576. }
  577. // checkUpSpam determine user can add up.
  578. func (s *Service) checkActionSpam(c context.Context, mid int64) (err error) {
  579. var ret int
  580. if ret, err = s.dao.Redis.SpamAction(c, mid); err != nil {
  581. log.Error("replyCacheDao.SpamAction(%d), err (%v)", mid, err)
  582. } else {
  583. if ret != ecode.OK.Code() {
  584. err = ecode.ReplyForbidAction
  585. }
  586. }
  587. return
  588. }
  589. // AddAction do act or cancel act for a reply.
  590. func (s *Service) AddAction(c context.Context, mid, oid, rpID int64, tp, action int8, ak, ck, op, platform, buvid string, build int64) (err error) {
  591. if err = reply.CheckAction(action); err != nil {
  592. return
  593. }
  594. user, err := s.userInfo(c, mid)
  595. if err != nil {
  596. return
  597. }
  598. if err = s.ValidUserAction(c, user); err != nil {
  599. return
  600. }
  601. if err = s.checkActionSpam(c, mid); err != nil {
  602. log.Error("s.checkActionSpam(%d) err (%v)", mid, err)
  603. return
  604. }
  605. r, err := s.reply(c, mid, oid, rpID, tp)
  606. if err != nil {
  607. return
  608. }
  609. // NOTE if the pending reply, the state is not normal
  610. if mid != r.Mid && !r.IsNormal() {
  611. err = ecode.ReplyForbidAction
  612. return
  613. }
  614. if s.RelationBlocked(c, r.Mid, mid) {
  615. err = ecode.ReplyBlacklistFilter
  616. return
  617. }
  618. var (
  619. userLikes map[int64]int8
  620. act int8
  621. )
  622. if userLikes, err = s.thumbup.HasLike(c, &thumdl.ArgHasLike{Business: "reply", MessageIDs: []int64{rpID}, Mid: mid}); err != nil {
  623. log.Error("s.thumbup.HasLike(%d,%d,%d) error(%v)", mid, rpID, oid, err)
  624. return
  625. }
  626. act = userLikes[rpID]
  627. now := time.Now()
  628. remoteIP := metadata.String(c, metadata.RemoteIP)
  629. var ac string
  630. if op == "like" {
  631. if (int8(act) == reply.ActionLike && action == reply.OpAdd) || (int8(act) != reply.ActionLike && action == reply.OpCancel) {
  632. err = ecode.ReplyActioned
  633. return
  634. }
  635. if action == reply.OpAdd {
  636. ac = reply.ReportReplyLike
  637. err = s.thumbup.Like(c, &thumdl.ArgLike{UpMid: r.Mid, Business: "reply", Mid: mid, MessageID: rpID, Type: thumdl.TypeLike, RealIP: remoteIP, OriginID: oid})
  638. } else {
  639. ac = reply.ReportReplyCancelLike
  640. err = s.thumbup.Like(c, &thumdl.ArgLike{UpMid: r.Mid, Business: "reply", Mid: mid, MessageID: rpID, Type: thumdl.TypeCancelLike, RealIP: remoteIP, OriginID: oid})
  641. }
  642. if err == nil {
  643. s.dao.Databus.Like(c, oid, rpID, mid, action, now.Unix())
  644. }
  645. } else {
  646. if (int8(act) == reply.ActionHate && action == reply.OpAdd) || (int8(act) != reply.ActionHate && action == reply.OpCancel) {
  647. err = ecode.ReplyActioned
  648. return
  649. }
  650. if action == reply.OpAdd {
  651. ac = reply.ReportReplyHate
  652. err = s.thumbup.Like(c, &thumdl.ArgLike{UpMid: r.Mid, Business: "reply", Mid: mid, MessageID: rpID, Type: thumdl.TypeDislike, RealIP: remoteIP, OriginID: oid})
  653. } else {
  654. ac = reply.ReportReplyCancelHate
  655. err = s.thumbup.Like(c, &thumdl.ArgLike{UpMid: r.Mid, Business: "reply", Mid: mid, MessageID: rpID, Type: thumdl.TypeCancelDislike, RealIP: remoteIP, OriginID: oid})
  656. }
  657. if err == nil {
  658. s.dao.Databus.Hate(c, oid, rpID, mid, action, now.Unix())
  659. }
  660. }
  661. if err != nil {
  662. if ecode.ThumbupCancelDislikeErr.Equal(err) || ecode.ThumbupCancelLikeErr.Equal(err) || ecode.ThumbupDupLikeErr.Equal(err) || ecode.ThumbupDupDislikeErr.Equal(err) {
  663. err = nil
  664. return
  665. }
  666. log.Error("thumbup (%d,%d,%d,%s,%d) failed!err:=%v", mid, oid, rpID, op, action, err)
  667. return
  668. }
  669. err = s.infoc.Info(mid, platform, build, buvid, 41, int(r.Type), r.Oid, ac, remoteIP, time.Now().Format("2006-01-02 15:04:05"), r.RpID, r.Mid, "", "", "", "", "")
  670. if err != nil {
  671. log.Error("infoc error (%v)", err)
  672. }
  673. return
  674. }
  675. func (s *Service) getIdsByRoots(c context.Context, oid int64, roots []int64, tp int8, pn, ps int) (sidsmap map[int64][]int64, ids []int64, err error) {
  676. var (
  677. start = (pn - 1) * ps
  678. end = start + ps - 1
  679. miss []int64
  680. tmprpIDs []int64
  681. )
  682. if sidsmap, ids, miss, err = s.dao.Redis.RangeByRoots(c, roots, start, end); err != nil {
  683. log.Error("s.dao.Redis.RangeByRoots() err(%v)", err)
  684. return
  685. }
  686. if len(miss) == 0 {
  687. return
  688. }
  689. for _, root := range miss {
  690. if tmprpIDs, err = s.dao.Reply.GetIdsByRoot(c, oid, root, tp, start, ps); err != nil {
  691. log.Error("s.dao.Reply.GetIdsByRoot(oid %d,tp %d,root %d) err(%v)", oid, tp, root, err)
  692. }
  693. if len(tmprpIDs) != 0 {
  694. sidsmap[root] = tmprpIDs
  695. ids = append(ids, tmprpIDs...)
  696. s.dao.Databus.RecoverIndexByRoot(c, oid, root, tp)
  697. }
  698. }
  699. return
  700. }
  701. func (s *Service) actions(c context.Context, mid, oid int64, rpIDs []int64) (amap map[int64]int8, err error) {
  702. if mid == 0 {
  703. amap = _emptyAction
  704. return
  705. }
  706. amap, err = s.thumbup.HasLike(c, &thumdl.ArgHasLike{Business: "reply", MessageIDs: rpIDs, Mid: mid})
  707. if err != nil {
  708. log.Error("s.thumbup.HasLike(%d, %d) error(%v)", mid, rpIDs, err)
  709. return
  710. }
  711. // NOTE: may have many keys
  712. // if mid not action,add -1 as mark
  713. if len(amap) == 0 {
  714. amap = map[int64]int8{-1: 0}
  715. }
  716. return
  717. }
  718. // getAccInfo get account infos of mids
  719. func (s *Service) getAccInfo(c context.Context, mids []int64) (cards map[int64]*accmdl.Card, err error) {
  720. if len(mids) == 0 {
  721. cards = _emptyCards
  722. return
  723. }
  724. var cardsReply *accmdl.CardsReply
  725. if cardsReply, err = s.acc.Cards3(c, &accmdl.MidsReq{Mids: mids}); err != nil {
  726. log.Error("s.acc.MultiInfo2(%v) error(%v)", mids, err)
  727. return nil, err
  728. }
  729. cards = cardsReply.Cards
  730. return
  731. }
  732. // GetBlacklist get account infos of mids
  733. func (s *Service) GetBlacklist(c context.Context, mid int64) (blacklistMap map[int64]bool, err error) {
  734. if mid == 0 {
  735. blacklistMap = _emptyBlackList
  736. return
  737. }
  738. var blacksReply *accmdl.BlacksReply
  739. if blacksReply, err = s.acc.Blacks3(c, &accmdl.MidReq{Mid: mid}); err != nil {
  740. log.Error("s.acc.Blacks(%v) error(%v)", mid, err)
  741. return
  742. }
  743. blacklistMap = blacksReply.BlackList
  744. return
  745. }
  746. // GetAttentions get relationships whether the user(mid) follows the target reply users
  747. func (s *Service) getAttentions(c context.Context, mid int64, targetMids []int64) (relations map[int64]*accmdl.RelationReply, err error) {
  748. if len(targetMids) == 0 {
  749. relations = _emptyRelations
  750. return
  751. }
  752. ip := metadata.String(c, metadata.RemoteIP)
  753. var relationsReply *accmdl.RelationsReply
  754. if relationsReply, err = s.acc.Relations3(c, &accmdl.RelationsReq{Mid: mid, Owners: targetMids, RealIp: ip}); err != nil {
  755. log.Error("s.acc.Relations2(%v, %v) error(%v)", mid, targetMids, err)
  756. return
  757. }
  758. relations = relationsReply.Relations
  759. return
  760. }
  761. // Subject get normal state reply subject
  762. func (s *Service) Subject(c context.Context, oid int64, tp int8) (*reply.Subject, error) {
  763. subject, err := s.getSubject(c, oid, tp)
  764. if err != nil {
  765. return nil, err
  766. }
  767. if subject.State == reply.SubStateForbid {
  768. return nil, ecode.ReplyForbidReply
  769. }
  770. return subject, nil
  771. }
  772. func (s *Service) getSubject(c context.Context, oid int64, tp int8) (*reply.Subject, error) {
  773. if !reply.LegalSubjectType(tp) {
  774. log.Error("illegal subject type: %v", tp)
  775. return nil, ecode.ReplyIllegalSubType
  776. }
  777. sub, err := s.dao.Mc.GetSubject(c, oid, tp)
  778. if err != nil {
  779. log.Error("replyCacheDao.GetSubject(%d, %d) error(%v)", oid, tp, err)
  780. }
  781. if sub != nil {
  782. return sub, nil
  783. }
  784. sub, err = s.dao.Subject.Get(c, oid, tp)
  785. if err != nil {
  786. log.Error("s.subject.Get(%d, %d) error(%v)", oid, tp, err)
  787. }
  788. if err == nil && sub != nil {
  789. s.dao.Mc.AddSubject(c, sub)
  790. return sub, nil
  791. }
  792. // fetch from remote call
  793. if tp != reply.SubTypeDrawyoo {
  794. log.Error("subject type is nether topic nor drawyoo: %v", tp)
  795. return nil, ecode.ReplyForbidReply
  796. }
  797. var mid int64
  798. if tp == reply.SubTypeDrawyoo {
  799. var yoo *drawyoo.Drawyoo
  800. if yoo, err = s.drawyoo.Info(c, oid); err != nil || yoo == nil {
  801. log.Warn("drawtyoo.DrawInfo(%d) not exist", oid)
  802. err = ecode.ReplyForbidReply
  803. return nil, err
  804. }
  805. mid = yoo.Mid
  806. }
  807. sub, err = s.upsertSubject(c, oid, tp, reply.SubStateNormal, mid)
  808. if err == ecode.ReplySubjectExist {
  809. sub, err = s.dao.Subject.Get(c, oid, tp)
  810. if err != nil {
  811. log.Error("s.subject.Get(%d, %d) error(%v)", oid, tp, err)
  812. return nil, err
  813. }
  814. return sub, nil
  815. }
  816. return sub, err
  817. }
  818. // upsertSubject insert or update a subject.
  819. func (s *Service) upsertSubject(c context.Context, oid int64, tp, state int8, mid int64) (sub *reply.Subject, err error) {
  820. now := time.Now()
  821. sub = &reply.Subject{
  822. Oid: oid,
  823. Type: tp,
  824. Mid: mid,
  825. State: state,
  826. CTime: xtime.Time(now.Unix()),
  827. MTime: xtime.Time(now.Unix()),
  828. }
  829. sub.ID, err = s.dao.Subject.Set(c, sub)
  830. if err != nil {
  831. log.Error("s.subject.Insert(%s) error(%v)", sub, err)
  832. return
  833. }
  834. if sub.ID == 0 {
  835. log.Warn("already have subject oid(%d) type(%d)", oid, tp)
  836. err = ecode.ReplySubjectExist
  837. }
  838. return
  839. }
  840. // setSubject insert or update a subject.
  841. //
  842. // Deprecated
  843. func (s *Service) setSubject(c context.Context, oid int64, tp, state int8, mid int64) (sub *reply.Subject, err error) {
  844. now := time.Now()
  845. sub, err = s.dao.Subject.Get(c, oid, tp)
  846. if err != nil {
  847. return
  848. }
  849. if sub != nil && sub.AttrVal(reply.SubAttrFrozen) == reply.AttrYes {
  850. err = ecode.ReplySubjectFrozen
  851. return
  852. }
  853. sub = &reply.Subject{
  854. Oid: oid,
  855. Type: tp,
  856. Mid: mid,
  857. State: state,
  858. CTime: xtime.Time(now.Unix()),
  859. MTime: xtime.Time(now.Unix()),
  860. }
  861. sub.ID, err = s.dao.Subject.Set(c, sub)
  862. if err != nil {
  863. log.Error("s.subject.Insert(%s) error(%v)", sub, err)
  864. return
  865. }
  866. if sub.ID == 0 {
  867. log.Warn("already have subject oid(%d) type(%d)", oid, tp)
  868. err = ecode.ReplySubjectExist
  869. }
  870. return
  871. }
  872. // Reply get reply from cache or db.
  873. // NOTE old php api call
  874. // TODO mobile jump
  875. func (s *Service) Reply(c context.Context, oid int64, tp int8, rpID int64) (r *reply.Reply, err error) {
  876. if r, err = s.GetReply(c, oid, rpID, tp); err != nil {
  877. log.Error("s.reply(oid %d,rpid %d) err(%v)", oid, rpID, err)
  878. return
  879. }
  880. r.Content, _ = s.dao.Content.Get(c, oid, rpID)
  881. arg := &accmdl.MidReq{Mid: r.Mid}
  882. r.Member = new(reply.Member)
  883. var card *accmdl.CardReply
  884. if card, err = s.acc.Card3(c, arg); err != nil {
  885. log.Error("s.acc.Info2(%d) error(%v)", r.Mid, err)
  886. return
  887. }
  888. r.Member.Info = &reply.Info{}
  889. if card != nil {
  890. r.Member.Info.FromCard(card.Card)
  891. }
  892. return
  893. }
  894. // Deprecated: use GetReply instead
  895. func (s *Service) reply(c context.Context, mid, oid, rpID int64, tp int8) (r *reply.Reply, err error) {
  896. if r, err = s.dao.Mc.GetReply(c, rpID); err != nil {
  897. log.Error("replyCacheDao.GetReply(%d, %d, %d) error(%v)", oid, rpID, tp, err)
  898. err = nil // NOTE ignore error
  899. }
  900. if r == nil {
  901. if r, err = s.dao.Reply.Get(c, oid, rpID); err != nil {
  902. log.Error("s.reply.GetReply(%d, %d) error(%v)", oid, rpID, err)
  903. return
  904. }
  905. }
  906. if r != nil {
  907. if r.Oid != oid || r.Type != tp {
  908. 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)
  909. err = ecode.RequestErr
  910. return
  911. }
  912. // NOTE if the pending reply, the state is audit
  913. if mid != r.Mid && !r.IsNormal() {
  914. err = ecode.ReplyNotExist
  915. return
  916. }
  917. } else {
  918. err = ecode.ReplyNotExist
  919. }
  920. return
  921. }
  922. // GetRootReply GetRootReply
  923. func (s *Service) GetRootReply(c context.Context, oid, rpID int64, tp int8) (*reply.Reply, error) {
  924. r, err := s.GetReply(c, oid, rpID, tp)
  925. if err != nil {
  926. return nil, err
  927. }
  928. if r.Root != 0 {
  929. return nil, ecode.ReplyIllegalRoot
  930. }
  931. return r, nil
  932. }
  933. // GetReply GetReply
  934. func (s *Service) GetReply(c context.Context, oid, rpID int64, tp int8) (*reply.Reply, error) {
  935. r, err := s.dao.Mc.GetReply(c, rpID)
  936. if err != nil {
  937. log.Error("replyCacheDao.GetReply(%d, %d, %d) error(%v)", oid, rpID, tp, err)
  938. err = nil // NOTE ignore error
  939. }
  940. if r == nil {
  941. r, err = s.dao.Reply.Get(c, oid, rpID)
  942. if err != nil {
  943. log.Error("s.reply.GetReply(%d, %d) error(%v)", oid, rpID, err)
  944. return nil, err
  945. }
  946. if r == nil {
  947. return nil, ecode.ReplyNotExist
  948. }
  949. if r.Oid != oid {
  950. 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)
  951. return nil, ecode.RequestErr
  952. }
  953. }
  954. return r, nil
  955. }
  956. // getReplyPos get root reply position.
  957. func (s *Service) getReplyPos(c context.Context, sub *reply.Subject, rp *reply.Reply) (pos int) {
  958. if ok, _ := s.dao.Redis.ExpireIndex(c, sub.Oid, sub.Type, reply.SortByFloor); ok {
  959. var err error
  960. if pos, err = s.dao.Redis.RankIndex(c, sub.Oid, sub.Type, rp.RpID, reply.SortByFloor); err == nil && pos >= 0 {
  961. pos++
  962. return
  963. }
  964. }
  965. // If get position from redis failed, then calc by subject
  966. pos = sub.Count - rp.Floor + 1
  967. return
  968. }
  969. // getReplyPosByRoot get reply position from root reply.
  970. func (s *Service) getReplyPosByRoot(c context.Context, rootRp *reply.Reply, rp *reply.Reply) (pos int) {
  971. if ok, _ := s.dao.Redis.ExpireIndexByRoot(c, rootRp.RpID); ok {
  972. var err error
  973. if pos, err = s.dao.Redis.RankIndexByRoot(c, rootRp.RpID, rp.RpID); err == nil && pos >= 0 {
  974. pos++
  975. return
  976. }
  977. }
  978. // If get position from redis failed, then calc by subject
  979. pos = rootRp.Count - rp.Floor + 1
  980. return
  981. }
  982. // Hide hide reply by upper.
  983. func (s *Service) Hide(c context.Context, oid, mid, rpID int64, tp int8, ak, ck string) (err error) {
  984. now := time.Now()
  985. if !reply.LegalSubjectType(tp) {
  986. err = ecode.ReplyIllegalSubType
  987. return
  988. }
  989. if !s.isUpper(c, mid, oid, tp) {
  990. err = ecode.AccessDenied
  991. return
  992. }
  993. if _, err = s.reply(c, mid, oid, rpID, tp); err != nil {
  994. return
  995. }
  996. s.dao.Databus.Hide(c, oid, rpID, tp, now.Unix())
  997. return
  998. }
  999. // Show show reply by upper.
  1000. func (s *Service) Show(c context.Context, oid, mid, rpID int64, tp int8, ak, ck string) (err error) {
  1001. now := time.Now()
  1002. if !reply.LegalSubjectType(tp) {
  1003. err = ecode.ReplyIllegalSubType
  1004. return
  1005. }
  1006. if !s.isUpper(c, mid, oid, tp) {
  1007. err = ecode.AccessDenied
  1008. return
  1009. }
  1010. if _, err = s.reply(c, mid, oid, rpID, tp); err != nil {
  1011. return
  1012. }
  1013. s.dao.Databus.Show(c, oid, rpID, tp, now.Unix())
  1014. return
  1015. }
  1016. // Emojis get vip emojis
  1017. func (s *Service) Emojis(c context.Context) (emo []*reply.EmojiPackage) {
  1018. emo = s.emojis
  1019. return
  1020. }
  1021. // UpperAddTop add top reply by upper
  1022. func (s *Service) UpperAddTop(c context.Context, mid, oid, rpID int64, tp, act int8, platform string, build int64, buvid string) (err error) {
  1023. var (
  1024. ts = time.Now().Unix()
  1025. r *reply.Reply
  1026. )
  1027. if !reply.LegalSubjectType(tp) {
  1028. err = ecode.ReplyIllegalSubType
  1029. return
  1030. }
  1031. if !s.isUpper(c, mid, oid, tp) {
  1032. err = ecode.AccessDenied
  1033. return
  1034. }
  1035. sub, err := s.Subject(c, oid, tp)
  1036. if err != nil {
  1037. log.Error("s.Subject(oid %v) err(%v)", oid, err)
  1038. return
  1039. }
  1040. if r, err = s.GetTop(c, sub, oid, tp, reply.ReplyAttrUpperTop); err != nil {
  1041. log.Error("s.GetTop(%d,%d) err(%v)", oid, tp, err)
  1042. return
  1043. }
  1044. if r != nil && act == 1 {
  1045. log.Warn("oid(%d) type(%d) already have top ", oid, tp)
  1046. err = ecode.ReplyHaveTop
  1047. return
  1048. }
  1049. if r == nil && act == 0 {
  1050. log.Warn("oid(%d) type(%d) do not have top ", oid, tp)
  1051. err = ecode.ReplyNotExist
  1052. return
  1053. }
  1054. if r != nil && r.RpID != rpID {
  1055. log.Error("reply not exist top(%v) rpID(%v)", r.RpID, rpID)
  1056. err = ecode.ReplyNotExist
  1057. return
  1058. }
  1059. // TODO: only need reply,no not need content and user info
  1060. if r, err = s.reply(c, mid, oid, rpID, tp); err != nil {
  1061. log.Error("s.GetReply err (%v)", err)
  1062. return
  1063. }
  1064. if r == nil {
  1065. log.Warn("oid(%d) type(%d) rpID(%d) do not exist ", oid, tp, rpID)
  1066. err = ecode.ReplyNotExist
  1067. return
  1068. }
  1069. if r.AttrVal(reply.ReplyAttrAdminTop) == 1 {
  1070. err = ecode.ReplyHaveTop
  1071. return
  1072. }
  1073. if r.Root != 0 {
  1074. log.Warn("oir(%d) type(%d) rpID(%d) not root reply", oid, tp, rpID)
  1075. err = ecode.ReplyNotRootReply
  1076. return
  1077. }
  1078. s.dao.Databus.UpperAddTop(c, mid, oid, rpID, ts, act, tp)
  1079. var action = reply.ReportReplyTop
  1080. if act == 0 {
  1081. action = reply.ReportReplyUntop
  1082. }
  1083. ip := metadata.String(c, metadata.RemoteIP)
  1084. report.User(&report.UserInfo{
  1085. Mid: r.Mid,
  1086. Platform: platform,
  1087. Build: build,
  1088. Buvid: buvid,
  1089. Business: 41,
  1090. Type: int(r.Type),
  1091. Oid: r.Oid,
  1092. Action: action,
  1093. Ctime: time.Now(),
  1094. IP: ip,
  1095. Index: []interface{}{
  1096. r.RpID,
  1097. },
  1098. })
  1099. return
  1100. }
  1101. // GetTop get upperTop reply from cache or db.
  1102. func (s *Service) GetTop(c context.Context, sub *reply.Subject, oid int64, tp int8, top uint32) (r *reply.Reply, err error) {
  1103. if (top == reply.ReplyAttrUpperTop) && sub.AttrVal(reply.SubAttrUpperTop) == 0 {
  1104. return
  1105. }
  1106. if (top == reply.ReplyAttrAdminTop) && sub.AttrVal(reply.SubAttrAdminTop) == 0 {
  1107. return
  1108. }
  1109. if r, err = s.dao.Mc.GetTop(c, oid, tp, top); err != nil {
  1110. log.Error("s.dao.Mc.GetAdminTop(%d, %d) error(%v)", oid, tp, err)
  1111. err = ecode.ServerErr
  1112. return
  1113. }
  1114. // NOTE load by job ,in case Cache penetration
  1115. if r == nil {
  1116. // if r, err = s.dao.Reply.GetTop(c, oid, tp, top); err != nil {
  1117. // log.Error("s.dao.Reply.GetTop(%d, %d) error(%v)", oid, tp, err)
  1118. // err = ecode.ServerErr
  1119. // return
  1120. // }
  1121. // if r == nil {
  1122. // err = ecode.ReplyNotExist
  1123. // return
  1124. // }
  1125. // if r.Content, err = s.dao.Content.Get(c, oid, r.rpID); err != nil {
  1126. // return
  1127. // }
  1128. // select {
  1129. // case s.topRpChan <- topRpChan{oid: oid, tp: tp, rp: r}:
  1130. // default:
  1131. // log.Warn("s.replyChan is full")
  1132. s.dao.Databus.AddTop(c, oid, tp, top)
  1133. }
  1134. return
  1135. }
  1136. // getRelation get account infos of mids
  1137. func (s *Service) getRelation(c context.Context, srcID, targetID int64, ip string) (uint32, error) {
  1138. if targetID == 0 {
  1139. return 0, nil
  1140. }
  1141. relMap, err := s.acc.RichRelations3(c, &accmdl.RichRelationReq{Owner: srcID, Mids: []int64{targetID}, RealIp: ip})
  1142. if err != nil || relMap == nil {
  1143. log.Error("s.acc.RichRelations2 sourceId(%v) targetId(%v)error(%v)", srcID, targetID, err)
  1144. // return normal relation if remote service is down!
  1145. return 0, nil
  1146. }
  1147. rel, ok := relMap.RichRelations[targetID]
  1148. if !ok {
  1149. // return normal relation if remote service is down!
  1150. return 0, nil
  1151. }
  1152. return relmdl.Attr(uint32(rel)), nil
  1153. }
  1154. // RelationBlocked RelationBlocked
  1155. func (s *Service) RelationBlocked(c context.Context, srcMid, targetMid int64) bool {
  1156. rel, _ := s.getRelation(c, srcMid, targetMid, "")
  1157. return rel == relmdl.AttrBlack
  1158. }
  1159. // SuperviseReply Supervise Reply
  1160. func (s *Service) SuperviseReply(c context.Context, mid int64, ak, ck string, tp int8) (err error) {
  1161. if conf.Conf.Supervision.Completed {
  1162. err = ecode.ReplyUpgrading
  1163. return
  1164. }
  1165. now := time.Now()
  1166. loc, _ := time.LoadLocation("Asia/Shanghai")
  1167. startTime, _ := time.ParseInLocation("2006-01-02 15:04:05", conf.Conf.Supervision.StartTime, loc)
  1168. endTime, _ := time.ParseInLocation("2006-01-02 15:04:05", conf.Conf.Supervision.EndTime, loc)
  1169. if now.Before(startTime) || now.After(endTime) {
  1170. err = nil
  1171. return
  1172. }
  1173. if tp == reply.SubTypeDrawyoo {
  1174. err = ecode.ReplyUpgrading
  1175. return
  1176. }
  1177. if overseas, _ := s.checkOverseasUser(c); overseas {
  1178. err = ecode.ReplyUpgrading
  1179. }
  1180. return
  1181. }
  1182. func (s *Service) checkOverseasUser(c context.Context) (overseas bool, err error) {
  1183. ip := metadata.String(c, metadata.RemoteIP)
  1184. arg := &locmdl.ArgIP{
  1185. IP: ip,
  1186. }
  1187. overseas = false
  1188. if respro, err := s.location.Info(c, arg); err != nil || respro == nil {
  1189. log.Error("s.location.Info(%s) error(%v) or respro is nil", ip, err)
  1190. } else {
  1191. if !strings.EqualFold(respro.Country, conf.Conf.Supervision.Location) {
  1192. overseas = true
  1193. }
  1194. }
  1195. return
  1196. }
  1197. // FetchFans fetching a fans relation array between upper(mid) and the users(uids)
  1198. func (s *Service) FetchFans(c context.Context, uids []int64, mid int64) (fans map[int64]*reply.FansDetail, err error) {
  1199. if fans, err = s.fans.Fetch(c, uids, mid, time.Now()); err != nil {
  1200. log.Error("s.fans.fetch(%d, %d) error(%v)", mid, uids, err)
  1201. }
  1202. return
  1203. }
  1204. // PaginateUpperDeletedLogs paginating the admin logs for size of 'pageSize', and returning the number of reporting, the number of admin logs delete by administrator
  1205. func (s *Service) PaginateUpperDeletedLogs(c context.Context, oid int64, tp int, curPage, pageSize int) (logs []*adminlog.AdminLog, replyCount, reportCount, pageCount, total int64, err error) {
  1206. var states = []int64{16, 18}
  1207. if logs, replyCount, reportCount, pageCount, total, err = s.search.LogPaginate(c, oid, tp, states, curPage, pageSize, conf.Conf.AssistConfig.StartTime, time.Now()); err != nil {
  1208. log.Error("s.adminlog.Paginate(%d, %d) error(%v)", oid, tp, err)
  1209. return nil, 0, 0, 0, 0, err
  1210. }
  1211. var mids = make([]int64, 0)
  1212. for _, d := range logs {
  1213. mids = append(mids, d.AdminID, d.ReplyMid)
  1214. }
  1215. minfos, err := s.getAccInfo(c, mids)
  1216. if err != nil {
  1217. log.Error("s.getAccInfo(mids %v) err(%v)", mids, err)
  1218. // NOTE degrade account
  1219. err = nil
  1220. }
  1221. for _, d := range logs {
  1222. if userInfo, ok := minfos[d.ReplyMid]; ok {
  1223. rs := []rune(userInfo.Name)
  1224. length := len(rs)
  1225. if length >= 3 {
  1226. d.ReplyUser = string(rs[0]) + "***" + string(rs[length-1])
  1227. } else if length == 2 {
  1228. d.ReplyUser = string(rs[0]) + "***"
  1229. } else {
  1230. d.ReplyUser = userInfo.Name
  1231. }
  1232. d.ReplyFacePic = userInfo.Face
  1233. }
  1234. if upperInfo, ok := minfos[d.AdminID]; ok {
  1235. d.Operator = upperInfo.Name
  1236. }
  1237. }
  1238. return
  1239. }
  1240. // GetReplyLogConfig get reply configuration from memocached or load a record from database by oid, type, category
  1241. func (s *Service) GetReplyLogConfig(c context.Context, sub *reply.Subject, category int8) (config *reply.Config, err error) {
  1242. if sub.AttrVal(reply.SubAttrConfig) == 0 {
  1243. return nil, nil
  1244. }
  1245. config, err = s.dao.Mc.GetReplyConfig(c, sub.Oid, sub.Type, category)
  1246. if err != nil {
  1247. log.Error("replyConfigCacheDao.GetReplyConfig(%d, %d, %d) error(%v)", sub.Oid, sub.Type, err)
  1248. err = nil // NOTE ignore error
  1249. }
  1250. if config == nil {
  1251. config, err = s.dao.Config.LoadConfig(c, sub.Oid, sub.Type, category)
  1252. if err != nil {
  1253. log.Error("s.reply.GetReply(%d, %d) error(%v)", sub.Oid, sub.Type, err)
  1254. return nil, err
  1255. }
  1256. if config == nil {
  1257. return nil, nil
  1258. }
  1259. if err = s.dao.Mc.AddReplyConfigCache(c, config); err != nil {
  1260. log.Error("replyConfigCacheDao.AddReplyConfig(%v) error(%v)", config, err)
  1261. return config, nil
  1262. }
  1263. }
  1264. return
  1265. }
  1266. // VerifyCaptcha VerifyCaptcha
  1267. func (s *Service) VerifyCaptcha(c context.Context, captcha string, mid int64) error {
  1268. token, err := s.dao.Mc.CaptchaToken(c, mid)
  1269. if err != nil {
  1270. return err
  1271. }
  1272. return s.dao.Captcha.Verify(c, token, captcha)
  1273. }
  1274. // Captcha return Captcha
  1275. func (s *Service) Captcha(c context.Context, mid int64) (string, error) {
  1276. token, uri, err := s.dao.Captcha.Captcha(c)
  1277. if err != nil {
  1278. return "", err
  1279. }
  1280. s.dao.Mc.SetCaptchaToken(c, mid, token)
  1281. return uri, nil
  1282. }
  1283. // Topics return topics
  1284. func (s *Service) Topics(c context.Context, mid int64, oid int64, typ int8, msg string) (topics []string, err error) {
  1285. if s.IsBnj(oid, typ) {
  1286. topics = []string{"拜年祭"}
  1287. return
  1288. }
  1289. topics, err = s.bigdata.Topics(c, mid, oid, typ, msg)
  1290. if err != nil {
  1291. return
  1292. }
  1293. if len(topics) == 0 {
  1294. return
  1295. }
  1296. messages := make(map[string]string)
  1297. for i := range topics {
  1298. key := strconv.FormatInt(int64(i), 10)
  1299. messages[key] = topics[i]
  1300. }
  1301. topics = topics[:0]
  1302. mf := &filgrpc.MFilterReq{
  1303. Area: "reply",
  1304. MsgMap: messages,
  1305. }
  1306. res, err := s.filcli.MFilter(c, mf)
  1307. if err != nil {
  1308. log.Error("s.fil.MFilter(%v) failed!err:=%v", messages, err)
  1309. return
  1310. }
  1311. for _, data := range res.RMap {
  1312. if data.Level > 15 {
  1313. continue
  1314. }
  1315. topics = append(topics, data.Result)
  1316. }
  1317. return
  1318. }
  1319. // IsHotReply IsHotReply
  1320. func (s *Service) IsHotReply(c context.Context, tp int8, oid, rpID int64) (isHot bool, err error) {
  1321. rpIDs, _, err := s.dao.Redis.Range(c, oid, tp, reply.SortByLike, 0, 5)
  1322. if err != nil {
  1323. log.Error("s.dao.Redis.Range() error(%v)", err)
  1324. return
  1325. }
  1326. rs, err := s.GetReplyByIDs(c, oid, tp, rpIDs)
  1327. if err != nil {
  1328. log.Error("s.GetReplyByIDs() error(%v)", err)
  1329. return
  1330. }
  1331. for _, rp := range rs {
  1332. if rpID == rp.RpID && rp.Like >= 3 {
  1333. isHot = true
  1334. return
  1335. }
  1336. }
  1337. return
  1338. }
  1339. // HotsBatch return HotsBatch
  1340. func (s *Service) HotsBatch(c context.Context, tp, size int8, oids []int64, mid int64) (res map[int64][]*reply.Reply, err error) {
  1341. var (
  1342. missed []int64
  1343. missedSubs []int64
  1344. m sync.Mutex
  1345. oidMap = make(map[int64][]int64)
  1346. subMap map[int64]*reply.Subject
  1347. )
  1348. res = make(map[int64][]*reply.Reply, len(oids))
  1349. if subMap, missedSubs, err = s.dao.Mc.GetMultiSubject(c, oids, tp); err != nil {
  1350. log.Error("s.dao.Mc.GetMultiSubject() error(%v)", err)
  1351. return
  1352. }
  1353. if len(missedSubs) > 0 {
  1354. var missedSubMap map[int64]*reply.Subject
  1355. if missedSubMap, err = s.dao.Subject.Gets(c, missedSubs, tp); err != nil {
  1356. log.Error("s.dao.Subject.Gets() error(%v)", err)
  1357. return
  1358. }
  1359. var subs []*reply.Subject
  1360. for oid, sub := range missedSubMap {
  1361. subMap[oid] = sub
  1362. subs = append(subs, sub)
  1363. }
  1364. s.cache.Do(c, func(ctx context.Context) { s.dao.Mc.AddSubject(ctx, subs...) })
  1365. }
  1366. if oidMap, missed, err = s.dao.Redis.RangeByOids(c, oids, tp, reply.SortByLike, 0, size); err != nil {
  1367. log.Error("s.dao.Redis.RangeByOids() error(%v)", err)
  1368. return
  1369. }
  1370. if len(missed) > 0 {
  1371. g, ctx := errgroup.WithContext(c)
  1372. for _, missedOid := range missed {
  1373. missedOid := missedOid
  1374. g.Go(func() error {
  1375. s.cache.Do(ctx, func(ctx context.Context) { s.dao.Databus.RecoverIndex(ctx, missedOid, tp, reply.SortByLike) })
  1376. rpIDs, err := s.dao.Reply.GetIdsSortLike(ctx, missedOid, tp, 0, int(size))
  1377. if err != nil {
  1378. return err
  1379. }
  1380. m.Lock()
  1381. oidMap[missedOid] = rpIDs
  1382. m.Unlock()
  1383. return nil
  1384. })
  1385. }
  1386. if err = g.Wait(); err != nil {
  1387. return
  1388. }
  1389. }
  1390. for oid, rpIDs := range oidMap {
  1391. var (
  1392. rpsMap map[int64]*reply.Reply
  1393. rps = make([]*reply.Reply, 0, len(rpIDs))
  1394. )
  1395. // 通过评论ID获取评论内容等元信息
  1396. if rpsMap, err = s.repliesMap(c, oid, tp, rpIDs); err != nil {
  1397. return
  1398. }
  1399. for _, rpID := range rpIDs {
  1400. if r, ok := rpsMap[rpID]; ok {
  1401. rps = append(rps, r)
  1402. }
  1403. }
  1404. if sub, ok := subMap[oid]; ok {
  1405. // 点赞以及关注等等关系构建
  1406. if err = s.buildReply(c, sub, rps, mid, false); err != nil {
  1407. return
  1408. }
  1409. }
  1410. res[oid] = rps
  1411. }
  1412. return
  1413. }