check.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460
  1. package service
  2. import (
  3. "context"
  4. "go-common/app/interface/main/creative/model/tag"
  5. "go-common/app/interface/main/videoup/model/archive"
  6. "go-common/app/interface/main/videoup/model/mission"
  7. "go-common/app/interface/main/videoup/model/porder"
  8. accapi "go-common/app/service/main/account/api"
  9. "go-common/library/ecode"
  10. "go-common/library/log"
  11. xtime "go-common/library/time"
  12. "go-common/library/xstr"
  13. "net/url"
  14. "regexp"
  15. "strings"
  16. "time"
  17. "unicode/utf8"
  18. )
  19. var (
  20. _emptyUnicodeRegForTitle = []*regexp.Regexp{
  21. regexp.MustCompile(`[\x{202e}]+`), // right-to-left override
  22. regexp.MustCompile(`[\x{200b}]+`), // zeroWithChar
  23. regexp.MustCompile(`[\x{1f6ab}]+`), // no_entry_sign
  24. regexp.MustCompile(`[\n]+`), // newline
  25. regexp.MustCompile(`[\r]+`), // newline
  26. }
  27. _emptyUnicodeReg = []*regexp.Regexp{
  28. regexp.MustCompile(`[\x{202e}]+`), // right-to-left override
  29. regexp.MustCompile(`[\x{200b}]+`), // zeroWithChar
  30. regexp.MustCompile(`[\x{1f6ab}]+`), // no_entry_sign
  31. }
  32. _nocharReg = []*regexp.Regexp{
  33. regexp.MustCompile(`[\p{Hangul}]+`), // kr
  34. regexp.MustCompile(`[\p{Tibetan}]+`), // tibe
  35. regexp.MustCompile(`[\p{Arabic}]+`), // arabic
  36. }
  37. _filenameReg = regexp.MustCompile(`^[A-Z0-9a-z]+$`) // only letter digital.
  38. _staffNameReg = regexp.MustCompile("^[\u4e00-\u9fa5a-zA-Z0-9]+$") //职能:数字、字母、中文
  39. )
  40. func (s *Service) checkMission(c context.Context, ap *archive.ArcParam) (err error) {
  41. if ap.MissionID <= 0 {
  42. log.Warn("MissionID(%d) error", ap.MissionID)
  43. ap.MissionID = 0
  44. return
  45. }
  46. missionID := ap.MissionID
  47. tid := ap.TypeID
  48. m, ok := s.missCache[missionID]
  49. if !ok || m.ID == 0 {
  50. err = ecode.VideoupMissionErr
  51. return
  52. }
  53. if m.ETime.Before(time.Now()) {
  54. log.Error("VideoupMissionEtimeInvalid err, tid(%d)|etime(%+v)", tid, m.ETime)
  55. err = ecode.VideoupMissionEtimeInvalid
  56. return
  57. }
  58. var missionTys map[int]*mission.Mission
  59. if missionTys, err = s.miss.MissionOnlineByTid(c, tid); err != nil {
  60. log.Error("MissionOnlineByTid err, s.tid(%+v)|err(%+v)", tid, err)
  61. err = nil
  62. return
  63. }
  64. // case: 这是活动全下架,还想参加活动
  65. if len(missionTys) == 0 {
  66. log.Error("missionTys already empty, tid(%d)|missionTys(%+v)", tid, missionTys)
  67. err = ecode.VideoupMissionNoMatch
  68. return
  69. }
  70. // 包含对分区无限制的活动
  71. if _, ok := missionTys[missionID]; !ok {
  72. log.Error("VideoupMissionNoMatch err, tid(%d)|missionID(%d)", tid, missionID)
  73. err = ecode.VideoupMissionNoMatch
  74. return
  75. }
  76. if ap.Copyright == archive.CopyrightCopy {
  77. log.Error("VideoupCopyForbidJoinMission err, copyright(%d)|missionID(%d)", ap.Copyright, ap.MissionID)
  78. err = ecode.VideoupCopyForbidJoinMission
  79. return
  80. }
  81. return
  82. }
  83. func (s *Service) checkMissionTag(srcTag string, missionID int) (dstTag string, err error) {
  84. var ctags = strings.Split(srcTag, ",")
  85. dstTag = srcTag
  86. // 交叉对比剔除掉当前活动已经使用的tag内容
  87. if len(s.missTagsCache) != 0 {
  88. var tags = make([]string, 0, len(ctags))
  89. for _, t := range ctags {
  90. if _, ok := s.missTagsCache[t]; !ok {
  91. tags = append(tags, t)
  92. }
  93. }
  94. dstTag = strings.Join(tags, ",")
  95. }
  96. // 校验
  97. // 两种情况报错提示用户: 1. 未参加活动,只提交一个tag,且是活动tag, 2. 参加活动,只提交了一个tag,且是其他活动的活动tag
  98. if dstTag == "" && missionID == 0 {
  99. log.Error("forbidMissionTagWithoutJoinMission srcTag(%s), MissionID(%d),s.missCache(%+v),s.missTagsCache(%+v)", srcTag, missionID, s.missCache, s.missTagsCache)
  100. err = ecode.VideoupTagForbidNotJoinMission
  101. return
  102. }
  103. // 未参加活动,就直接返回校验后的tag内容
  104. m, ok := s.missCache[missionID]
  105. if !ok {
  106. return
  107. }
  108. // 如果参加了当前有效的活动,就会把对应的活动第一个tag拼接在头部, 对于活动id和tag不匹配的会做校验
  109. var singleMissionTag string
  110. if m.Tags != "" {
  111. singleMissionTag = strings.Split(m.Tags, ",")[0]
  112. } else {
  113. singleMissionTag = m.Name
  114. }
  115. if len(dstTag) > 0 {
  116. dstTag = singleMissionTag + "," + dstTag
  117. } else {
  118. dstTag = singleMissionTag
  119. }
  120. return
  121. }
  122. func (s *Service) checkVideo(ap *archive.ArcParam) (err error) {
  123. vds := make([]*archive.VideoParam, 0)
  124. fnMap := make(map[string]int)
  125. var ok bool
  126. for i, v := range ap.Videos {
  127. if v == nil {
  128. continue
  129. }
  130. if v.Title, ok = s.checkTitle(v.Title); !ok {
  131. newErr := ecode.VideoupVideoTitleErr
  132. err = ecode.Errorf(newErr, newErr.Message(), i+1)
  133. log.Error("ap.Videos checkTitle err(%+v)|Title(%s)", err, v.Title)
  134. return
  135. }
  136. if v.Desc, ok = s.checkDesc(v.Desc); !ok {
  137. newErr := ecode.VideoupVideoDescErr
  138. err = ecode.Errorf(newErr, newErr.Message(), i+1)
  139. log.Error("ap.Videos checkDesc err(%+v)|Desc(%s)", err, v.Desc)
  140. return
  141. }
  142. if ok = _filenameReg.MatchString(v.Filename); !ok {
  143. newErr := ecode.VideoupVideoFilenameErr
  144. err = ecode.Errorf(newErr, newErr.Message(), i+1)
  145. log.Error("ap.Videos _filenameReg err(%+v)|filename(%s)", err, v.Filename)
  146. return
  147. }
  148. if v.Cid == 0 && v.Filename == "" { // NOTE: cid>0 means code mode
  149. newErr := ecode.VideoupVideoFilenameErr
  150. err = ecode.Errorf(newErr, newErr.Message(), i+1)
  151. log.Error("ap.Videos err(%+v)|Filename(%s)|Cid(%d)", err, v.Filename, v.Cid)
  152. return
  153. }
  154. if _, ok := fnMap[v.Filename]; ok {
  155. err = ecode.VideoupFilenameCanotRepeat
  156. log.Error("ecode.VideoupFilenameCanotRepeat err(%+v)|Filename(%s)|index(%d)", err, v.Filename, i)
  157. return
  158. }
  159. vds = append(vds, v)
  160. fnMap[v.Filename] = 1
  161. }
  162. ap.Videos = vds
  163. return
  164. }
  165. func (s *Service) checkCover(cover string) (cv string, ok bool) {
  166. if cover == "" {
  167. ok = true
  168. return
  169. }
  170. uri, err := url.Parse(cover)
  171. if err != nil {
  172. return
  173. }
  174. if strings.Contains(uri.Host, "hdslb.com") {
  175. cv = uri.Path
  176. ok = true
  177. return
  178. } else if strings.Contains(uri.Host, "acgvideo.com") {
  179. cv = cover
  180. ok = true
  181. return
  182. }
  183. return
  184. }
  185. func (s *Service) checkDynamicLen233(dynamic string) (dyn string, ok bool) {
  186. dyn = strings.TrimSpace(dynamic)
  187. var _emptyDynUnicodeReg = []*regexp.Regexp{
  188. regexp.MustCompile(`[\x{FFFC}]+`), // obj
  189. }
  190. for _, reg := range _emptyDynUnicodeReg {
  191. dyn = reg.ReplaceAllString(dyn, "")
  192. }
  193. if utf8.RuneCountInString(dyn) > 233 {
  194. return
  195. }
  196. ok = true
  197. return
  198. }
  199. func (s *Service) checkTitle(title string) (ct string, ok bool) {
  200. ct = strings.TrimSpace(title)
  201. if utf8.RuneCountInString(ct) > 80 {
  202. return
  203. }
  204. for _, reg := range _nocharReg {
  205. if reg.MatchString(ct) {
  206. return
  207. }
  208. }
  209. for _, reg := range _emptyUnicodeRegForTitle {
  210. ct = reg.ReplaceAllString(ct, "")
  211. }
  212. ok = true
  213. return
  214. }
  215. func (s *Service) checkDesc(desc string) (cd string, ok bool) {
  216. cd = strings.TrimSpace(desc)
  217. for _, reg := range _emptyUnicodeReg {
  218. cd = reg.ReplaceAllString(cd, "")
  219. }
  220. if utf8.RuneCountInString(cd) > 2000 {
  221. return
  222. }
  223. ok = true
  224. return
  225. }
  226. func (s *Service) checkAccount(c context.Context, mid int64, ip string) (p *accapi.Profile, err error) {
  227. if p, err = s.acc.Profile(c, mid, ip); err != nil {
  228. return
  229. }
  230. if p.Silence == 1 {
  231. err = ecode.UserDisabled
  232. } else if p.Level < 1 {
  233. err = ecode.UserLevelLow
  234. }
  235. if _, ok := s.exemptZeroLevelAndAnswerUps[mid]; ok && err == ecode.UserLevelLow {
  236. log.Info("s.exemptZeroLevelAndAnswerUps, (%s),(%d),(%+v)", ip, mid, err)
  237. err = nil
  238. }
  239. return
  240. }
  241. func (s *Service) checkOrderID(c context.Context, mid, orderID int64, ip string) (err error) {
  242. orderIDs, err := s.order.ExecuteOrders(c, mid, ip)
  243. if err != nil {
  244. log.Error("s.order.ExecuteOrders mid(%d) ip(%s) error(%v)", mid, err)
  245. err = ecode.VideoupOrderAPIErr
  246. return
  247. }
  248. if _, ok := orderIDs[orderID]; !ok {
  249. err = ecode.VideoupOrderIDNotAllow
  250. }
  251. return
  252. }
  253. func (s *Service) checkIdentify(c context.Context, mid int64, ip string) (err error) {
  254. if _, ok := s.exemptIDCheckUps[mid]; ok {
  255. log.Info("s.exemptIDCheckUps, (%s),(%d),(%+v)", ip, mid, err)
  256. return
  257. }
  258. // fault-tolerant for service interruption
  259. if err = s.acc.IdentifyInfo(c, ip, mid); err != nil {
  260. if err != ecode.UserCheckNoPhone && err != ecode.UserCheckInvalidPhone {
  261. log.Warn("s.accIdentifyInfo, account service maybe in interruption,(%s),(%d),(%+v)", ip, mid, err)
  262. return nil
  263. }
  264. log.Error("s.accIdentifyInfo, (%s),(%d),(%+v)", ip, mid, err)
  265. return
  266. }
  267. return
  268. }
  269. // checkPorderForAdd
  270. func (s *Service) checkPorderForAdd(c context.Context, ap *archive.ArcParam, mid int64) (err error) {
  271. // 防止脏数据, 强制计算和校验Porder的数据
  272. ap.Porder.FlowID = 1
  273. log.Info("ap.Porder (%+v)", ap.Porder)
  274. // showType check
  275. if len(ap.Porder.ShowType) > 0 {
  276. var showTypes []int64
  277. if showTypes, err = xstr.SplitInts(ap.Porder.ShowType); err != nil {
  278. log.Error("SplitInts ShowType err, (%s),(%+v)", ap.Porder.ShowType, err)
  279. return
  280. }
  281. //广告的展现形式太多或者太少
  282. if len(showTypes) == 0 {
  283. err = ecode.VideoupAdShowTypeErr
  284. log.Error("check showTypes (%+v)|err(%+v)", ap.Porder, err)
  285. return
  286. }
  287. for _, showType := range showTypes {
  288. if showType > 0 {
  289. if _, ok := s.PorderCfgs[showType]; !ok {
  290. err = ecode.VideoupAdShowTypeErr
  291. log.Error("VideoupAdShowTypeErr Porder(%+v)|err(%+v)", ap.Porder.ShowType, err)
  292. return
  293. }
  294. }
  295. }
  296. }
  297. // Official check
  298. if ap.Porder.Official == 1 {
  299. if _, ok := porder.OfficialIndustryMaps[ap.Porder.IndustryID]; !ok {
  300. err = ecode.VideoupAdOfficialIndustryIDErr
  301. log.Error("VideoupAdOfficialIndustryIDErr Porder(%+v)|err(%+v)", ap.Porder, err)
  302. return
  303. }
  304. if ap.Porder.BrandID < 0 {
  305. err = ecode.VideoupAdBrandIDErr
  306. log.Error("VideoupAdBrandIDErr Porder(%+v)|err(%+v)", ap.Porder, err)
  307. return
  308. }
  309. // logic map to OfficialIndustryMaps waiting for add other official industry
  310. if _, ok := s.PorderGames[ap.Porder.BrandID]; !ok {
  311. err = ecode.VideoupAdBrandIDErr
  312. log.Error("VideoupAdBrandIDErr Porder(%+v)|err(%+v)", ap.Porder, err)
  313. return
  314. }
  315. }
  316. // Industry check
  317. if ap.Porder.IndustryID > 0 {
  318. if _, ok := s.PorderCfgs[ap.Porder.IndustryID]; !ok {
  319. err = ecode.VideoupAdIndustryIDErr
  320. log.Error("VideoupAdIndustryIDErr Porder(%+v)|err(%+v)", ap.Porder, err)
  321. return
  322. }
  323. }
  324. return
  325. }
  326. // checkDescForLength fn
  327. func (s *Service) checkDescForLength(desc string, descFormatID int, typeID int16, copyright int8) (err error) {
  328. if descFormatID == 0 {
  329. if utf8.RuneCountInString(desc) > 250 {
  330. err = ecode.VideoupFmDesLenOverLimit
  331. log.Error("ecode.VideoupFmDesLenOverLimit, desc(%s),formatID(%d)", desc, descFormatID)
  332. return
  333. }
  334. return
  335. }
  336. if utf8.RuneCountInString(desc) > 2000 {
  337. err = ecode.VideoupFmDesLenOverLimit
  338. log.Error("ecode.VideoupFmDesLenOverLimit, desc(%s),formatID(%d)", desc, descFormatID)
  339. return
  340. }
  341. return
  342. }
  343. // checkVideos, 1: check len(video) == ? 0 , 2: check typeID in (145,146,147,83) and with multi Videos
  344. func (s *Service) checkVideos(c context.Context, ap *archive.ArcParam) (err error) {
  345. if len(ap.Videos) == 0 {
  346. log.Error("checkVideos vds length 0")
  347. err = ecode.VideoupZeroVideos
  348. return
  349. }
  350. if len(ap.Videos) > 1 && ap.ForbidMultiVideoType() {
  351. log.Error("checkVideos vds ForbidMultiVideoType, len(%d), type(%d) ", len(ap.Videos), ap.TypeID)
  352. err = ecode.VideoupForbidMultiVideoForTypes
  353. return
  354. }
  355. return
  356. }
  357. // tagsCheck fn
  358. func (s *Service) tagsCheck(c context.Context, mid int64, tagName, ip string) (err error) {
  359. var t *tag.Tag
  360. tags := strings.Split(tagName, ",")
  361. for i, tagStr := range tags {
  362. if t, err = s.tag.TagCheck(c, mid, tagStr); err != nil {
  363. log.Error("s.tag.TagCheck(%d, %+v, %s) error(%+v)", mid, t, ip, err)
  364. err = nil
  365. return
  366. }
  367. if t != nil && (t.State == tag.TagStateDel || t.State == tag.TagStateHide || t.Type == tag.OfficailActiveTag) {
  368. newErr := ecode.VideoupTagForbid
  369. err = ecode.Errorf(newErr, newErr.Message(), i+1)
  370. log.Error("s.tag.VideoupTagForbid (%d, %+v, %s) error(%+v)", mid, t, ip, err)
  371. return
  372. }
  373. }
  374. return
  375. }
  376. // checkVideosMaxLimitForEdit fn
  377. func (s *Service) checkVideosMaxLimitForEdit(vs []*archive.Video, pvideos []*archive.VideoParam) (addCnt int) {
  378. fnMaps := make(map[string]string)
  379. for _, v := range vs {
  380. fnMaps[v.Filename] = v.Filename
  381. }
  382. for _, v := range pvideos {
  383. if _, exist := fnMaps[v.Filename]; len(v.Filename) > 0 && !exist {
  384. addCnt++
  385. }
  386. }
  387. return
  388. }
  389. // checkPay fn
  390. func (s *Service) checkAddPay(c context.Context, ap *archive.ArcParam, ip string) (err error) {
  391. if err = s.checkPayProtocol(c, ap.Pay, ap.Mid); err != nil {
  392. log.Error("s.checkAddPayProtocol (ap %+v) error(%+v)", ap, err)
  393. return
  394. }
  395. if err = s.checkPayLimit(c, ap); err != nil {
  396. log.Error("s.checkAddPayLimit (ap %+v) error(%+v)", ap, err)
  397. return
  398. }
  399. if err = s.checkPayWithOrder(c, ap.Porder, ap.Pay, ap.OrderID, ap.Mid); err != nil {
  400. log.Error("s.checkAddPayWithOrder (ap %+v) error(%+v)", ap, err)
  401. return
  402. }
  403. return
  404. }
  405. // checkEditPay fn
  406. // 关于RefuseUpdate
  407. // 1.当ctime==ptime的时候,也就是尚未稿件一审,都是可以update付费模块的
  408. // 2.开放过的,打回的稿件可以在任何时间点进行付费修改
  409. // 3.开放过的,非打回的稿件想要修改只能等60天时间过了之后(时间锁是为了保护普通用户的收看权利),否则自己申诉付费审核人员,要求强制打回
  410. func (s *Service) checkEditPay(c context.Context, ap *archive.ArcParam, a *archive.Archive, ip string) (err error) {
  411. _, registed, _ := s.pay.Ass(c, a.Aid, ip)
  412. // 只要注册过付费信息都允许自主修改付费模块
  413. if registed {
  414. if err = s.checkPayProtocol(c, ap.Pay, ap.Mid); err != nil {
  415. log.Error("s.checkAddPayProtocol (ap %+v) error(%+v)", ap, err)
  416. return
  417. }
  418. if err = s.checkPayLimit(c, ap); err != nil {
  419. log.Error("s.checkAddPayLimit (ap %+v) error(%+v)", ap, err)
  420. return
  421. }
  422. if err = s.checkPayWithOrder(c, ap.Porder, ap.Pay, ap.OrderID, ap.Mid); err != nil {
  423. log.Error("s.checkAddPayWithOrder (ap %+v) error(%+v)", ap, err)
  424. return
  425. }
  426. //如果其他端都不传付费信息,那么就用现在有的来进行最后覆盖
  427. if ap.Pay == nil {
  428. ap.UgcPay = a.UgcPay
  429. return
  430. }
  431. updateDeadLine := xtime.Time(a.PTime.Time().AddDate(0, 0, s.c.UgcPayAllowEditDays).Unix())
  432. if ap.Pay != nil &&
  433. a.CTime != a.PTime &&
  434. a.State != archive.StateForbidRecicle &&
  435. xtime.Time(time.Now().Unix()) < updateDeadLine {
  436. log.Warn("checkEditPay ap.Pay.RefuseUpdate updateDeadLine (%+v)|(%+v)|(%+v)", a.Aid, a.CTime, a.PTime)
  437. ap.Pay.RefuseUpdate = true
  438. }
  439. }
  440. return
  441. }