public.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547
  1. package service
  2. import (
  3. "context"
  4. "fmt"
  5. "go-common/app/interface/main/creative/conf"
  6. "go-common/app/interface/main/creative/dao/account"
  7. "go-common/app/interface/main/creative/dao/activity"
  8. "go-common/app/interface/main/creative/dao/archive"
  9. "go-common/app/interface/main/creative/dao/article"
  10. "go-common/app/interface/main/creative/dao/creative"
  11. "go-common/app/interface/main/creative/dao/pay"
  12. "go-common/app/interface/main/creative/dao/subtitle"
  13. "net/url"
  14. "os"
  15. "go-common/app/interface/main/creative/dao/tag"
  16. "go-common/app/interface/main/creative/dao/up"
  17. actmdl "go-common/app/interface/main/creative/model/activity"
  18. arcinter "go-common/app/interface/main/creative/model/archive"
  19. arcmdl "go-common/app/interface/main/creative/model/archive"
  20. "go-common/app/interface/main/creative/model/music"
  21. mMdl "go-common/app/interface/main/creative/model/music"
  22. "go-common/app/interface/main/creative/model/newcomer"
  23. tagMdl "go-common/app/interface/main/creative/model/tag"
  24. accMdl "go-common/app/service/main/account/model"
  25. mdlarc "go-common/app/service/main/archive/model/archive"
  26. "go-common/library/log"
  27. "go-common/library/queue/databus"
  28. "go-common/library/sync/errgroup"
  29. xtime "go-common/library/time"
  30. "go-common/library/xstr"
  31. "hash/crc32"
  32. "math"
  33. "strconv"
  34. "strings"
  35. "time"
  36. )
  37. //Public struct
  38. type Public struct {
  39. c *conf.Config
  40. creative *creative.Dao
  41. sub *subtitle.Dao
  42. acc *account.Dao
  43. act *activity.Dao
  44. arc *archive.Dao
  45. up *up.Dao
  46. pay *pay.Dao
  47. tag *tag.Dao
  48. // type cache
  49. TypesCache map[string][]*arcmdl.Type
  50. TopTypesCache []*arcmdl.Type
  51. TypeMapCache map[int16]*arcmdl.Type
  52. CTypesCache map[string][]*arcmdl.Type
  53. AllMusics map[int64]*mMdl.Music
  54. DescFmtsCache map[int64]map[int8]map[int8]*arcmdl.DescFormat
  55. DescFmtsArrCache []*arcmdl.DescFormat
  56. // cache
  57. ActVideoAllCache []*actmdl.Activity
  58. TopActCache []*actmdl.Activity
  59. ActMapCache map[int64]*actmdl.Activity
  60. StaffTitlesCache []*tagMdl.StaffTitle
  61. //task
  62. taskPub *databus.Databus
  63. AppWhiteMidsByGroups map[int64]map[int64]int64
  64. }
  65. //RPCDaos struct
  66. type RPCDaos struct {
  67. Arc *archive.Dao
  68. Acc *account.Dao
  69. Art *article.Dao
  70. Up *up.Dao
  71. Sub *subtitle.Dao
  72. }
  73. //New get service
  74. func New(c *conf.Config, rpcdaos *RPCDaos) *Public {
  75. p := &Public{
  76. c: c,
  77. creative: creative.New(c),
  78. arc: archive.New(c),
  79. sub: subtitle.New(c),
  80. act: activity.New(c),
  81. pay: pay.New(c),
  82. tag: tag.New(c),
  83. acc: rpcdaos.Acc,
  84. up: rpcdaos.Up,
  85. taskPub: databus.New(c.TaskPub),
  86. AllMusics: make(map[int64]*mMdl.Music),
  87. ActVideoAllCache: make([]*actmdl.Activity, 0),
  88. TopActCache: make([]*actmdl.Activity, 0),
  89. StaffTitlesCache: make([]*tagMdl.StaffTitle, 0),
  90. ActMapCache: make(map[int64]*actmdl.Activity),
  91. AppWhiteMidsByGroups: make(map[int64]map[int64]int64),
  92. }
  93. p.loadTypes()
  94. p.loadDescFormat()
  95. p.loadMusicTable()
  96. p.loadActivities()
  97. p.loadPortalGroups()
  98. p.loadStaffTitles()
  99. go p.loadproc()
  100. go p.tableproc()
  101. return p
  102. }
  103. //loadPortalGroups fn
  104. func (p *Public) loadPortalGroups() {
  105. var (
  106. tmpGroupMaps = make(map[int64]map[int64]int64)
  107. specialGroupIDs map[int64]int8
  108. )
  109. if os.Getenv("DEPLOY_ENV") == "uat" {
  110. specialGroupIDs = map[int64]int8{
  111. 29: 1,
  112. 12: 1,
  113. }
  114. } else {
  115. specialGroupIDs = map[int64]int8{
  116. 22: 1, // 移动端新手任务白名单
  117. 23: 1, // OPG用户组(内部人员名单
  118. }
  119. }
  120. c := context.TODO()
  121. gps := make([]int64, 0)
  122. for gpKey := range specialGroupIDs {
  123. gps = append(gps, gpKey)
  124. }
  125. type ChData struct {
  126. gp int64
  127. gmap map[int64]int64
  128. }
  129. rechan := make(chan ChData, len(gps))
  130. g, ctx := errgroup.WithContext(c)
  131. for _, gpID := range gps {
  132. var gid = gpID
  133. g.Go(func() error {
  134. ret, e := p.up.UpSpecial(ctx, gid)
  135. if e != nil {
  136. log.Warn("p.up.UpSpecial gid (%d)", gid)
  137. return nil
  138. }
  139. if len(ret) > 0 {
  140. log.Warn("len of ret gid (%d)|(%d)", gid, len(ret))
  141. }
  142. rechan <- ChData{gid, ret}
  143. return nil
  144. })
  145. }
  146. g.Wait()
  147. close(rechan)
  148. for c := range rechan {
  149. tmpGroupMaps[c.gp] = c.gmap
  150. }
  151. p.AppWhiteMidsByGroups = tmpGroupMaps
  152. }
  153. func (p *Public) tableproc() {
  154. for {
  155. time.Sleep(time.Duration(10 * time.Second))
  156. p.loadMusicTable()
  157. }
  158. }
  159. func (p *Public) loadMusicTable() {
  160. var (
  161. err error
  162. musicMap map[int64]*mMdl.Music
  163. )
  164. c := context.TODO()
  165. if musicMap, err = p.arc.AllMusics(c); err != nil {
  166. log.Error("p.music.MCategorys err(%+v)", err)
  167. return
  168. }
  169. if musicMap != nil {
  170. p.AllMusics = musicMap
  171. }
  172. log.Info("loadMusicTable (%d)", len(p.AllMusics))
  173. }
  174. // NewRPCDaos get all
  175. func NewRPCDaos(c *conf.Config) *RPCDaos {
  176. rds := &RPCDaos{
  177. Arc: archive.New(c),
  178. Acc: account.New(c),
  179. Art: article.New(c),
  180. Up: up.New(c),
  181. Sub: subtitle.New(c),
  182. }
  183. return rds
  184. }
  185. // loadproc
  186. func (p *Public) loadproc() {
  187. for {
  188. time.Sleep(5 * time.Minute)
  189. p.loadTypes()
  190. p.loadDescFormat()
  191. p.loadActivities()
  192. p.loadPortalGroups()
  193. p.loadStaffTitles()
  194. }
  195. }
  196. // loadActivities fn
  197. func (p *Public) loadActivities() {
  198. p.ActVideoAllCache = make([]*actmdl.Activity, 0)
  199. videoallActs, err := p.act.Activities(context.TODO())
  200. if err != nil {
  201. return
  202. }
  203. for _, act := range videoallActs {
  204. if len(act.Tags) == 0 {
  205. act.Tags = act.Name
  206. } else {
  207. act.Tags = strings.Split(act.Tags, ",")[0]
  208. }
  209. v := &actmdl.Activity{
  210. ID: act.ID,
  211. Name: act.Name,
  212. Tags: act.Tags,
  213. ActURL: act.ActURL,
  214. Protocol: act.Protocol,
  215. Type: act.Type,
  216. Hot: act.Hot,
  217. STime: act.STime,
  218. }
  219. p.ActVideoAllCache = append(p.ActVideoAllCache, v)
  220. p.ActMapCache[act.ID] = v
  221. }
  222. topLen := 4
  223. multiplier := p.c.Coefficient.ActHeat
  224. if len(p.ActVideoAllCache) <= topLen {
  225. p.TopActCache = p.ActVideoAllCache
  226. } else {
  227. p.TopActCache = p.ActVideoAllCache[:topLen]
  228. }
  229. for _, topAct := range p.TopActCache {
  230. stime, _ := time.Parse("2006-01-02 15:04:05", topAct.STime)
  231. stimeAfter3Day := stime.AddDate(0, 0, 3).Unix()
  232. if time.Now().Unix() < stimeAfter3Day {
  233. topAct.New = 1
  234. }
  235. likeCnt, _ := p.act.Likes(context.Background(), topAct.ID)
  236. if likeCnt > 0 {
  237. topAct.Comment = fmt.Sprintf("%d人参与", int(math.Ceil(float64(likeCnt)*multiplier)))
  238. }
  239. }
  240. }
  241. //load types
  242. func (p *Public) loadTypes() {
  243. tops, langs, typeMap, err := p.creative.Types(context.TODO())
  244. if err != nil {
  245. log.Error("p.creative.Types error(%v)", err)
  246. return
  247. }
  248. arcmdl.SortRulesForTopTypes(tops, arcmdl.WebType)
  249. p.TopTypesCache = tops
  250. for _, vals := range langs {
  251. arcmdl.SortRulesForTopTypes(vals, arcmdl.WebType)
  252. }
  253. p.TypesCache = langs
  254. p.CTypesCache = genCTypesCache(langs)
  255. p.TypeMapCache = typeMap
  256. }
  257. // 自动过滤不需要的二级分区,如果二级分区全部删除了,自动会删除对应的一级分区
  258. func genCTypesCache(langs map[string][]*arcmdl.Type) (CTypesCache map[string][]*arcmdl.Type) {
  259. CTypesCache = make(map[string][]*arcmdl.Type)
  260. for lang, topTypes := range langs {
  261. CTypesCache[lang] = make([]*arcmdl.Type, 0)
  262. for _, topType := range topTypes {
  263. nt := &arcmdl.Type{
  264. ID: topType.ID,
  265. Lang: topType.Lang,
  266. Parent: topType.Parent,
  267. Name: topType.Name,
  268. Desc: topType.Desc,
  269. Descapp: topType.Descapp,
  270. Count: topType.Count,
  271. Original: topType.Original,
  272. IntroCopy: topType.IntroCopy,
  273. Notice: topType.Notice,
  274. CopyRight: topType.CopyRight,
  275. Show: topType.Show,
  276. Rank: topType.Rank,
  277. Children: []*arcmdl.Type{},
  278. }
  279. if arcmdl.ForbidTopTypesForAppAdd(topType.ID) {
  280. nt.Show = false
  281. }
  282. for _, child := range topType.Children {
  283. if arcmdl.ForbidSubTypesForAppAdd(child.ID) {
  284. continue
  285. }
  286. nt.Children = append(nt.Children, child)
  287. }
  288. if len(nt.Children) > 0 {
  289. CTypesCache[lang] = append(CTypesCache[lang], nt)
  290. }
  291. }
  292. arcmdl.SortRulesForTopTypes(CTypesCache[lang], arcmdl.AppType)
  293. }
  294. return
  295. }
  296. // CoverURL convert cover url to full url.
  297. func CoverURL(uri string) (cover string) {
  298. if uri == "" {
  299. //cover = "http://static.hdslb.com/images/transparent.gif"
  300. return
  301. }
  302. cover = uri
  303. if strings.Index(uri, "http://") == 0 {
  304. return
  305. }
  306. if len(uri) >= 10 && uri[:10] == "/templets/" {
  307. return
  308. }
  309. if strings.HasPrefix(uri, "group1") {
  310. cover = "http://i0.hdslb.com/" + uri
  311. return
  312. }
  313. if pos := strings.Index(uri, "/uploads/"); pos != -1 && (pos == 0 || pos == 3) {
  314. cover = uri[pos+8:]
  315. }
  316. cover = strings.Replace(cover, "{IMG}", "", -1)
  317. cover = "http://i" + strconv.FormatInt(int64(crc32.ChecksumIEEE([]byte(cover)))%3, 10) + ".hdslb.com" + cover
  318. return
  319. }
  320. //BatchArchives batch get archive info.
  321. func (p *Public) BatchArchives(c context.Context, mid int64, aids []int64, ip string) (avm map[int64]*arcmdl.ArcVideo, err error) {
  322. avm, err = p.arc.Views(c, mid, aids, ip)
  323. if err != nil {
  324. log.Error("p.arc.Views aids (%v), ip(%s) err(%v)", aids, ip, err)
  325. }
  326. return
  327. }
  328. func (p *Public) loadDescFormat() {
  329. fmts, err := p.arc.DescFormat(context.TODO())
  330. if err != nil {
  331. return
  332. }
  333. fmtsArr := make([]*arcmdl.DescFormat, 0)
  334. tp := make(map[int64]map[int8]map[int8]*arcmdl.DescFormat)
  335. for _, d := range fmts {
  336. fmtsArr = append(fmtsArr, d)
  337. if _, okTp := tp[d.TypeID]; !okTp {
  338. tp[d.TypeID] = make(map[int8]map[int8]*arcmdl.DescFormat)
  339. }
  340. if _, okCp := tp[d.TypeID][d.Copyright]; !okCp {
  341. tp[d.TypeID][d.Copyright] = make(map[int8]*arcmdl.DescFormat)
  342. }
  343. if _, okCp := tp[d.TypeID][d.Copyright][d.Lang]; !okCp {
  344. tp[d.TypeID][d.Copyright][d.Lang] = &arcmdl.DescFormat{}
  345. }
  346. tp[d.TypeID][d.Copyright][d.Lang] = d
  347. }
  348. p.DescFmtsCache = tp
  349. p.DescFmtsArrCache = fmtsArr
  350. }
  351. //TaskPub fn pub task finished msg.
  352. func (p *Public) TaskPub(mid int64, from, count int) (err error) {
  353. msg := &newcomer.TaskMsg{
  354. MID: mid,
  355. From: from,
  356. Count: int64(count),
  357. TimeStamp: time.Now().Unix(),
  358. }
  359. log.Info("task Pub mid(%d) msg(%+v)", mid, msg)
  360. if err = p.taskPub.Send(context.TODO(), strconv.FormatInt(mid, 10), msg); err != nil {
  361. log.Error("s.taskPub.Send mid(%d) error(%v)", mid, err)
  362. return
  363. }
  364. return
  365. }
  366. // StaffList fn
  367. func (p *Public) StaffList(c context.Context, aid int64, cache bool) (res []*arcinter.Staff, err error) {
  368. if cache {
  369. return p.arc.StaffData(c, aid)
  370. }
  371. if res, err = p.arc.RawStaffData(c, aid); err != nil {
  372. log.Error("s.StaffList(%d) error(%v)", aid, err)
  373. return
  374. }
  375. return
  376. }
  377. //BgmBindList fn
  378. func (p *Public) BgmBindList(c context.Context, aid, cid, mType int64, cache bool) (resOk []*arcinter.ViewBGM, err error) {
  379. var (
  380. data *creative.BgmData
  381. newIDS, sids []int64
  382. ret map[int64]string
  383. musics map[int64]*music.Music
  384. res []*arcinter.ViewBGM
  385. )
  386. //无更新逻辑 注意空缓存
  387. if data, err = p.creative.BgmData(c, aid, cid, mType, cache); err != nil || data == nil {
  388. log.Error("s.GetMaterialData(%d,%d,%d) error(%v)", aid, cid, mType, err)
  389. return
  390. }
  391. if sids, err = xstr.SplitInts(data.Data); err != nil {
  392. log.Error("s.BgmBindList(%d,%d,%d) error(%v)", aid, cid, mType, err)
  393. return
  394. }
  395. for _, sid := range sids {
  396. if sid > 0 {
  397. newIDS = append(newIDS, sid)
  398. }
  399. }
  400. if len(newIDS) < 1 {
  401. return
  402. }
  403. //all localcache
  404. musics = p.AllMusics
  405. if musics == nil {
  406. return
  407. }
  408. res = make([]*arcinter.ViewBGM, 0)
  409. resOk = make([]*arcinter.ViewBGM, 0)
  410. var mids []int64
  411. for _, sid := range newIDS {
  412. if _, ok := musics[sid]; !ok {
  413. continue
  414. }
  415. musicData := musics[sid]
  416. newOne := &arcinter.ViewBGM{}
  417. newOne.SID = sid
  418. newOne.MID = musicData.UpMID
  419. newOne.Title = musicData.Name
  420. newOne.Author = musicData.Musicians
  421. if musicData.State == 0 {
  422. params := url.Values{}
  423. params.Set("bgm_id", strconv.FormatInt(sid, 10))
  424. params.Set("from_aid", strconv.FormatInt(aid, 10))
  425. params.Set("from_cid", strconv.FormatInt(cid, 10))
  426. params.Set("from_source", "player_page")
  427. newOne.JumpURL = p.c.H5Page.Cooperate + "?" + params.Encode()
  428. }
  429. mids = append(mids, musicData.UpMID)
  430. res = append(res, newOne)
  431. }
  432. if ret, err = p.getUpNames(c, mids); err != nil {
  433. log.Error("s.BgmBindList(%d,%d,%d) get mid(%v) name error(%v)", aid, cid, mType, mids, err)
  434. err = nil
  435. }
  436. for _, v := range res {
  437. if name, ok := ret[v.MID]; ok {
  438. v.Author = name
  439. }
  440. resOk = append(resOk, v)
  441. }
  442. if len(resOk) > 5 {
  443. resOk = resOk[:5]
  444. }
  445. return
  446. }
  447. // getUpNames fn
  448. func (p *Public) getUpNames(c context.Context, mids []int64) (ret map[int64]string, err error) {
  449. var (
  450. minfos map[int64]*accMdl.Info
  451. )
  452. ret = make(map[int64]string)
  453. if len(mids) > 0 {
  454. minfos, err = p.acc.Infos(c, mids, "localhost")
  455. if err != nil {
  456. log.Info("minfos err mids (%+v)|err(%+v)", mids, err)
  457. return
  458. }
  459. for _, info := range minfos {
  460. ret[info.Mid] = info.Name
  461. }
  462. }
  463. return
  464. }
  465. // FillPayInfo fill pay
  466. func (p *Public) FillPayInfo(c context.Context, a *arcmdl.Archive, ugcPayCfg *conf.UgcPay, ip string) (pay *arcmdl.UgcPayInfo) {
  467. var (
  468. err error
  469. ass *arcmdl.PayAsset
  470. registed bool
  471. )
  472. pay = &arcmdl.UgcPayInfo{
  473. Acts: make(map[string]*arcmdl.PayAct),
  474. }
  475. pay.Acts["edit"] = &arcmdl.PayAct{
  476. State: 1,
  477. }
  478. pay.Acts["delete"] = &arcmdl.PayAct{
  479. State: 1,
  480. }
  481. ass, registed, err = p.pay.Ass(c, a.Aid, ip)
  482. if err != nil {
  483. log.Error("p.pay.Ass aids (%v), ip(%s) err(%v)", a.Aid, ip, err)
  484. }
  485. pay.Asset = ass
  486. delDeadline := xtime.Time(a.PTime.Time().AddDate(0, 0, ugcPayCfg.AllowDeleteDays).Unix())
  487. editDeadline := xtime.Time(a.PTime.Time().AddDate(0, 0, ugcPayCfg.AllowEditDays).Unix())
  488. if !registed {
  489. pay.Acts["edit"] = &arcmdl.PayAct{
  490. Reason: "老稿件不允许参与UGC内容付费项目中,请重新投稿",
  491. State: 0,
  492. }
  493. } else {
  494. if a.UgcPay == 1 {
  495. if a.CTime != a.PTime &&
  496. xtime.Time(time.Now().Unix()) < delDeadline {
  497. pay.Acts["delete"] = &arcmdl.PayAct{
  498. Reason: fmt.Sprintf("付费稿件必须在开放之后的第%d天才能删除", ugcPayCfg.AllowDeleteDays),
  499. State: 0,
  500. }
  501. }
  502. if a.CTime != a.PTime &&
  503. a.State != mdlarc.StateForbidRecicle &&
  504. xtime.Time(time.Now().Unix()) < editDeadline {
  505. pay.Acts["edit"] = &arcmdl.PayAct{
  506. Reason: fmt.Sprintf("付费稿件必须在开放之后的第%d天才能编辑", ugcPayCfg.AllowDeleteDays),
  507. State: 0,
  508. }
  509. }
  510. }
  511. // 有注册过,但是现在已经被关闭付费标记的稿件,也允许编辑和删除
  512. }
  513. return
  514. }
  515. // loadStaffTitles 拉取联合投稿职能列表
  516. func (p *Public) loadStaffTitles() {
  517. var (
  518. c = context.TODO()
  519. err error
  520. )
  521. if p.StaffTitlesCache, err = p.tag.StaffTitleList(c); err != nil {
  522. log.Error("p.loadStaffTitles() error(%v)", err)
  523. return
  524. }
  525. }