dao.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. package faq
  2. import (
  3. "context"
  4. "encoding/json"
  5. "fmt"
  6. "go-common/app/interface/main/creative/conf"
  7. "go-common/app/interface/main/creative/model/faq"
  8. "go-common/library/cache/redis"
  9. "go-common/library/ecode"
  10. "go-common/library/log"
  11. bm "go-common/library/net/http/blademaster"
  12. "go-common/library/net/metadata"
  13. xtime "go-common/library/time"
  14. "net/url"
  15. "strconv"
  16. )
  17. const (
  18. _notRobot = -1
  19. _rsOk = "000000"
  20. _searchURL = "/kb/searchInerDocListBilibili/4"
  21. _hdKey = "faq_%s_%d_%d_%d"
  22. )
  23. // Dao define
  24. type Dao struct {
  25. c *conf.Config
  26. client *bm.Client
  27. searchURL string
  28. redis *redis.Pool
  29. redisExpire int32
  30. }
  31. // New init dao
  32. func New(c *conf.Config) (d *Dao) {
  33. d = &Dao{
  34. c: c,
  35. client: bm.NewClient(c.HTTPClient.Normal),
  36. searchURL: c.Host.HelpAPI + _searchURL,
  37. redis: redis.NewPool(c.Redis.Cover.Config),
  38. redisExpire: int32(120),
  39. }
  40. return
  41. }
  42. func keyHd(qTypeID string, keyFlag, pn, ps int) string {
  43. return fmt.Sprintf(_hdKey, qTypeID, keyFlag, pn, ps)
  44. }
  45. // SetDetailCache set help detail to cache.
  46. func (d *Dao) SetDetailCache(c context.Context, qTypeID string, keyFlag, pn, ps, total int, data []*faq.Detail) (err error) {
  47. conn := d.redis.Get(c)
  48. defer conn.Close()
  49. count := 0
  50. key := keyHd(qTypeID, keyFlag, pn, ps)
  51. if err = conn.Send("DEL", key); err != nil {
  52. log.Error("conn.Send(DEL, %s) error(%v)", key, err)
  53. return
  54. }
  55. count++
  56. var bs []byte
  57. for _, detail := range data {
  58. if bs, err = json.Marshal(detail); err != nil {
  59. log.Error("json.Marshal(%v) error (%v)", detail, err)
  60. return
  61. }
  62. if err = conn.Send("ZADD", key, combineHd(detail.UpdateTime, total), bs); err != nil {
  63. log.Error("conn.Send(ZADD, %s, %s) error(%v)", key, string(bs), err)
  64. return
  65. }
  66. count++
  67. }
  68. if err = conn.Send("EXPIRE", key, d.redisExpire); err != nil {
  69. log.Error("conn.Send(Expire, %s, %d) error(%v)", key, d.redisExpire, err)
  70. return
  71. }
  72. count++
  73. if err = conn.Flush(); err != nil {
  74. log.Error("conn.Flush error(%v)", err)
  75. return
  76. }
  77. for i := 0; i < count; i++ {
  78. if _, err = conn.Receive(); err != nil {
  79. log.Error("conn.Receive() error(%v)", err)
  80. return
  81. }
  82. }
  83. return
  84. }
  85. // DetailCache get help detail to cache.
  86. func (d *Dao) DetailCache(c context.Context, qTypeID string, keyFlag, pn, ps int) (res []*faq.Detail, count int, err error) {
  87. conn := d.redis.Get(c)
  88. defer conn.Close()
  89. key := keyHd(qTypeID, keyFlag, pn, ps)
  90. values, err := redis.Values(conn.Do("ZREVRANGE", key, 0, -1, "WITHSCORES"))
  91. if err != nil {
  92. log.Error("conn.Do(ZREVRANGE, %s) error(%v)", key, err)
  93. return
  94. }
  95. if len(values) == 0 {
  96. return
  97. }
  98. if err = conn.Flush(); err != nil {
  99. log.Error("conn.Flush() err(%v)", err)
  100. return
  101. }
  102. var num int64
  103. for len(values) > 0 {
  104. bs := []byte{}
  105. if values, err = redis.Scan(values, &bs, &num); err != nil {
  106. log.Error("redis.Scan(%v) error(%v)", values, err)
  107. return
  108. }
  109. detail := &faq.Detail{}
  110. if err = json.Unmarshal(bs, detail); err != nil {
  111. log.Error("json.Unmarshal(%v) error(%v)", bs, err)
  112. return
  113. }
  114. res = append(res, detail)
  115. }
  116. count = fromHd(num)
  117. return
  118. }
  119. func fromHd(i int64) int {
  120. return int(i & 0xffff)
  121. }
  122. func combineHd(create xtime.Time, count int) int64 {
  123. return create.Time().Unix()<<16 | int64(count)
  124. }
  125. // Detail fn
  126. func (d *Dao) Detail(c context.Context, qTypeID string, keyFlag, pn, ps int) (data []*faq.Detail, total int, err error) {
  127. ip := metadata.String(c, metadata.RemoteIP)
  128. params := url.Values{}
  129. params.Set("questionTypeId", qTypeID)
  130. params.Set("keyFlag", strconv.Itoa(keyFlag))
  131. params.Set("keyWords", "")
  132. params.Set("pageNo", strconv.Itoa(pn))
  133. params.Set("pageSize", strconv.Itoa(ps))
  134. params.Set("robotFlag", strconv.Itoa(_notRobot))
  135. var res struct {
  136. Code string `json:"retCode"`
  137. Data []*faq.Detail `json:"items"`
  138. Total int `json:"totalCount"`
  139. }
  140. if err = d.client.Get(c, d.searchURL, ip, params, &res); err != nil {
  141. log.Error("Detail d.searchURL url(%s)|(%+v)", d.searchURL+"?"+params.Encode(), res)
  142. err = ecode.HelpDetailError
  143. return
  144. }
  145. log.Info("Detail d.searchURL url(%s)|(%+v)", d.searchURL+"?"+params.Encode(), res.Code)
  146. if res.Code != _rsOk {
  147. log.Error("Detail d.searchURL url(%s)|(%+v)", d.searchURL+"?"+params.Encode(), res.Code)
  148. err = ecode.HelpDetailError
  149. return
  150. }
  151. total = res.Total
  152. data = res.Data
  153. return
  154. }