service.go 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. package service
  2. import (
  3. "bytes"
  4. "context"
  5. "crypto/md5"
  6. "crypto/sha256"
  7. "encoding/base64"
  8. "encoding/hex"
  9. "fmt"
  10. "net/url"
  11. "strconv"
  12. "strings"
  13. "time"
  14. hmdl "go-common/app/interface/main/history/model"
  15. hrpc "go-common/app/interface/main/history/rpc/client"
  16. "go-common/app/interface/main/report-click/conf"
  17. "go-common/app/interface/main/report-click/dao"
  18. "go-common/app/interface/main/report-click/service/crypto/aes"
  19. "go-common/app/interface/main/report-click/service/crypto/padding"
  20. accmdl "go-common/app/service/main/account/model"
  21. accrpc "go-common/app/service/main/account/rpc/client"
  22. "go-common/library/ecode"
  23. "go-common/library/log"
  24. "go-common/library/net/metadata"
  25. "go-common/library/stat/prom"
  26. "go-common/library/sync/pipeline/fanout"
  27. )
  28. // 0 1 2 3 4 5 6 7 8 9 : <-> d w o i k p s x m q l
  29. // 48 49 50 51 52 53 54 55 56 57 58 <-> 100 119 111 105 107 112 115 120 109 113 108
  30. var ecKeys = map[rune]rune{
  31. 48: 119,
  32. 49: 111,
  33. 50: 105,
  34. 51: 107,
  35. 52: 112,
  36. 53: 115,
  37. 54: 120,
  38. 55: 109,
  39. 56: 113,
  40. 57: 108,
  41. 58: 100,
  42. 119: 48,
  43. 111: 49,
  44. 105: 50,
  45. 107: 51,
  46. 112: 52,
  47. 115: 53,
  48. 120: 54,
  49. 109: 55,
  50. 113: 56,
  51. 108: 57,
  52. 100: 58,
  53. }
  54. // Service service struct info.
  55. type Service struct {
  56. c *conf.Config
  57. d *dao.Dao
  58. accRPC *accrpc.Service3
  59. hisRPC *hrpc.Service
  60. cache *fanout.Fanout
  61. promErr *prom.Prom
  62. promInfo *prom.Prom
  63. }
  64. // New service.
  65. func New(c *conf.Config) (s *Service) {
  66. s = &Service{
  67. c: c,
  68. d: dao.New(c),
  69. accRPC: accrpc.New3(c.AccRPC),
  70. hisRPC: hrpc.New(c.HisRPC),
  71. cache: fanout.New("cache", fanout.Worker(1), fanout.Buffer(1024)),
  72. promErr: prom.BusinessErrCount,
  73. promInfo: prom.BusinessInfoCount,
  74. }
  75. return
  76. }
  77. // FlashSigned flash Signed.
  78. func (s *Service) FlashSigned(params url.Values, secret string, now time.Time) (err error) {
  79. st := params.Get("stime")
  80. stime, err := strconv.ParseInt(st, 10, 64)
  81. if err != nil {
  82. err = ecode.ClickQueryFormatErr
  83. return
  84. }
  85. if now.Unix()-stime > 60 {
  86. err = ecode.ClickServerTimeout
  87. return
  88. }
  89. sign := params.Get("sign")
  90. params.Del("sign")
  91. mh := md5.Sum([]byte(strings.ToLower(params.Encode()) + secret))
  92. if hex.EncodeToString(mh[:]) != sign {
  93. err = ecode.ClickQuerySignErr
  94. }
  95. return
  96. }
  97. // Decrypt decrypt bytes by aes key and iv.
  98. func (s *Service) Decrypt(src []byte, aesKey, aesIv string) (res []byte, err error) {
  99. res, err = aes.CBCDecrypt(src, []byte(aesKey), []byte(aesIv), padding.PKCS5)
  100. if err != nil {
  101. log.Error("aes.CBCDecrypt(%s, %s, %s) error(%v)", base64.StdEncoding.EncodeToString(src), s.c.Click.AesKey, s.c.Click.AesIv, err)
  102. err = ecode.ClickAesDecryptErr
  103. }
  104. return
  105. }
  106. // Verify verify bytes from post body.
  107. func (s *Service) Verify(src []byte, aesSalt string, now time.Time) (p url.Values, err error) {
  108. p, err = url.ParseQuery(string(src))
  109. if err != nil {
  110. err = ecode.ClickQueryFormatErr
  111. return
  112. }
  113. // check server time
  114. st := p.Get("stime")
  115. stime, err := strconv.ParseInt(st, 10, 64)
  116. if err != nil {
  117. err = ecode.ClickQueryFormatErr
  118. return
  119. }
  120. if now.Unix()-stime > 60*3 {
  121. err = ecode.ClickServerTimeout
  122. return
  123. }
  124. // verify sign
  125. sign := p.Get("sign")
  126. sbs, err := hex.DecodeString(sign)
  127. if err != nil {
  128. log.Error("hex.DecodeString(%s) error(%v)", sign, err)
  129. err = ecode.ClickQuerySignErr
  130. return
  131. }
  132. p.Del("sign")
  133. // sha 256
  134. h := sha256.New()
  135. // h.Write([]byte(strings.ToLower(p.Encode())))
  136. h.Write([]byte(p.Encode()))
  137. h.Write([]byte(aesSalt))
  138. bs := h.Sum(nil)
  139. // bytes queal
  140. if !bytes.Equal(sbs, bs) {
  141. log.Error("hmac.Equal(%s, %x) params(%s) not equal", sign, bs, p.Encode())
  142. err = ecode.ClickHmacSignErr
  143. }
  144. return
  145. }
  146. // Play send play count to kafka.
  147. func (s *Service) Play(c context.Context, plat, aid, cid, part, mid, level, ftime, stime, did, ip, agent, buvid, cookieSid, refer, typeID, subType, sid, epid, playMode, platform, device, mobiAapp, autoPlay, session string) {
  148. if aid == "" || aid == "0" {
  149. return
  150. }
  151. m, errP := strconv.ParseInt(mid, 10, 64)
  152. if errP != nil {
  153. log.Warn("strconv.ParseInt(%s) error(%v)", mid, errP)
  154. mid = "0"
  155. }
  156. if m != 0 {
  157. arg := &accmdl.ArgMid{Mid: m}
  158. res, err := s.accRPC.Card3(c, arg)
  159. if err != nil {
  160. log.Error("s.accRPC.UserInfo() error(%v)", err)
  161. return
  162. }
  163. if res.Silence == 1 {
  164. log.Warn("user mid(%d) spacesta(%d) too lower", m, res.Silence)
  165. return
  166. }
  167. level = fmt.Sprintf("%d", res.Level)
  168. }
  169. s.cache.Do(c, func(ctx context.Context) {
  170. s.d.Play(ctx, plat, aid, cid, part, mid, level, ftime, stime, did, ip, agent, buvid, cookieSid, refer, typeID, subType, sid, epid, playMode, platform, device, mobiAapp, autoPlay, session)
  171. })
  172. }
  173. // GenDid gen did.
  174. func (s *Service) GenDid(ip string, now time.Time) string {
  175. var src string
  176. ft := now.Unix() - int64(now.Second())
  177. uip, ok := parseIP(ip)
  178. if uip == nil {
  179. return ""
  180. }
  181. if ok {
  182. src = fmt.Sprintf("%d:%d", netAtoN(uip), ft)
  183. return myEncryptDecrypt(src)
  184. }
  185. fs := encode(uint64(ft))
  186. ipRes := ipv6AtoN(uip)
  187. if len(ipRes) > 25 { // total 32, 25 for ip, 1 for :, 6 for ftime
  188. ipRes = ipRes[:25]
  189. }
  190. return fmt.Sprintf("%s:%s", ipRes, fs)
  191. }
  192. // CheckDid check did.
  193. func (s *Service) CheckDid(did string) (ip, ft string) {
  194. params := strings.Split(did, ":")
  195. if len(params) == 4 {
  196. log.Warn("report click did:%s", did)
  197. return ntoIPv6(params[:3]), fmt.Sprintf("%d", decode([]byte(params[3])))
  198. }
  199. dst := myEncryptDecrypt(did)
  200. params = strings.Split(dst, ":")
  201. if len(params) != 2 {
  202. return
  203. }
  204. ipInt, _ := strconv.ParseInt(params[0], 10, 64)
  205. ip = netNtoA(uint32(ipInt))
  206. ft = params[1]
  207. return
  208. }
  209. func myEncryptDecrypt(src string) (dst string) {
  210. var tmp []rune
  211. for _, k := range src {
  212. if _, ok := ecKeys[k]; !ok {
  213. return ""
  214. }
  215. tmp = append(tmp, ecKeys[k])
  216. }
  217. dst = string(tmp)
  218. return
  219. }
  220. // Report report to history.
  221. func (s *Service) Report(c context.Context, proStr, cidStr, tpStr, subType, realtimeStr, aidStr, midstr, sidStr, epidStr, dtStr, tsStr string) (err error) {
  222. var (
  223. tp, stp, dt int
  224. )
  225. if tp, err = strconv.Atoi(tpStr); err != nil {
  226. log.Warn("Report type:%s", tpStr)
  227. }
  228. stp, _ = strconv.Atoi(subType)
  229. mid, _ := strconv.ParseInt(midstr, 10, 64)
  230. aid, _ := strconv.ParseInt(aidStr, 10, 64)
  231. sid, _ := strconv.ParseInt(sidStr, 10, 64)
  232. epid, _ := strconv.ParseInt(epidStr, 10, 64)
  233. cid, _ := strconv.ParseInt(cidStr, 10, 64)
  234. if aid == 0 && cid == 0 {
  235. return ecode.RequestErr
  236. }
  237. pro, _ := strconv.ParseInt(proStr, 10, 64)
  238. realtime, _ := strconv.ParseInt(realtimeStr, 10, 64)
  239. if dt, err = strconv.Atoi(dtStr); err != nil {
  240. dt = 2
  241. }
  242. ip := metadata.String(c, metadata.RemoteIP)
  243. ts, _ := strconv.ParseInt(tsStr, 10, 64)
  244. history := &hmdl.History{Aid: aid, Sid: sid, Epid: epid, TP: int8(tp), STP: int8(stp), Cid: cid, DT: int8(dt), Pro: pro, Unix: ts}
  245. arg := &hmdl.ArgHistory{Mid: mid, Realtime: realtime, RealIP: ip, History: history}
  246. return s.hisRPC.Add(c, arg)
  247. }