feedback.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646
  1. package service
  2. import (
  3. "context"
  4. "net"
  5. "sort"
  6. "strconv"
  7. "strings"
  8. "time"
  9. "unicode/utf8"
  10. "go-common/app/interface/main/feedback/model"
  11. locmdl "go-common/app/service/main/location/model"
  12. "go-common/library/database/sql"
  13. "go-common/library/ecode"
  14. "go-common/library/log"
  15. "go-common/library/net/metadata"
  16. xtime "go-common/library/time"
  17. "go-common/library/xstr"
  18. )
  19. const (
  20. _androidFeedback = "android-feedback"
  21. _androidFeedbackI = "android-fb-i"
  22. _iosFeedback = "ios-feedback"
  23. _iosFeedbackI = "ios-fb-i"
  24. _androidPlayer = "android-player"
  25. _iosPlayer = "ios-player"
  26. _androidCreative = "android-creative"
  27. _iosCreative = "ios-creative"
  28. _androidLivePink = "android-live-pink"
  29. _iosLivePink = "ios-live-pink"
  30. _tvYST = "tv-yst"
  31. _creativeCenter = "creative-center"
  32. )
  33. var (
  34. defaultTag = &model.Tag{
  35. ID: 314,
  36. Name: "其它问题",
  37. Type: 0,
  38. }
  39. )
  40. // AddReply add feedback raply and create session if session isn't exist
  41. func (s *Service) AddReply(c context.Context, mid, tagID int64, buvid, system, version, mobiApp, content, imgURL, logURL, device, channel, entrance,
  42. netState, netOperator, agencyArea, platform, browser, qq, email string, now time.Time) (r *model.Reply, err error) {
  43. // check size
  44. if utf8.RuneCountInString(content) > s.c.Feedback.MaxContentSize {
  45. err = ecode.FeedbackBodyTooLarge
  46. return
  47. }
  48. var (
  49. ssn *model.Session
  50. reply *model.Reply
  51. cTime xtime.Time
  52. replyID = buvid
  53. id int64
  54. imgURLs []string
  55. ip = metadata.String(c, metadata.RemoteIP)
  56. )
  57. if entrance == "view" {
  58. if model.IsAndroid(model.Plat(mobiApp, device)) {
  59. platform = _androidPlayer
  60. } else if model.IsIOS(model.Plat(mobiApp, device)) {
  61. platform = _iosPlayer
  62. }
  63. } else if entrance == _tvYST || entrance == _creativeCenter {
  64. platform = entrance
  65. } else if platform == "android" {
  66. if entrance == _androidCreative {
  67. platform = _androidCreative
  68. } else if entrance == _androidLivePink {
  69. platform = _androidLivePink
  70. } else if model.Plat(mobiApp, device) == model.PlatAndroid {
  71. platform = _androidFeedback
  72. } else if model.Plat(mobiApp, device) == model.PlatAndroidI {
  73. platform = _androidFeedbackI
  74. }
  75. } else if platform == "ios" {
  76. if entrance == _iosCreative {
  77. platform = _iosCreative
  78. } else if entrance == _iosLivePink {
  79. platform = _iosLivePink
  80. } else if model.Plat(mobiApp, device) == model.PlatIPhone {
  81. platform = _iosFeedback
  82. } else if model.Plat(mobiApp, device) == model.PlatIPhoneI {
  83. platform = _iosFeedbackI
  84. }
  85. }
  86. if ssn, err = s.dao.Session(c, buvid, system, version, mid); err != nil {
  87. log.Error("s.dao.Session(%s,%s,%s,%d) error(%v)", buvid, system, version, mid, err)
  88. return
  89. }
  90. if mid != 0 {
  91. replyID = strconv.FormatInt(mid, 10)
  92. }
  93. cTime = xtime.Time(now.Unix())
  94. if ssn == nil {
  95. if content != "" || logURL != "" || imgURL != "" {
  96. // TODO delete ;
  97. imgURLs = strings.Split(imgURL, ";")
  98. var sid int64
  99. sid, err = s.session(c, mid, tagID, buvid, system, version, "", content, imgURLs[0], logURL, device, channel, ip, entrance, netState, netOperator, agencyArea, platform, browser, qq, email, now)
  100. if err == nil {
  101. for _, v := range imgURLs[1:] {
  102. if v != "" {
  103. reply = &model.Reply{
  104. SessionID: sid,
  105. ReplyID: replyID,
  106. Content: "",
  107. ImgURL: v,
  108. LogURL: "",
  109. CTime: cTime,
  110. MTime: cTime,
  111. }
  112. if id, err = s.dao.AddReply(c, reply); err != nil {
  113. log.Error("s.dao.AddReply error(%v)", err)
  114. continue
  115. }
  116. }
  117. }
  118. }
  119. }
  120. } else {
  121. if entrance == "view" || entrance == _androidLivePink || entrance == _iosLivePink || platform == _androidLivePink || platform == _iosLivePink || model.IsPlayerScreen(tagID) || entrance == _tvYST || platform == _tvYST || entrance == _creativeCenter || platform == _creativeCenter {
  122. _, err = s.session(c, mid, tagID, buvid, system, version, "", content, imgURL, logURL, device, channel, ip, entrance, netState, netOperator, agencyArea, platform, browser, qq, email, now)
  123. if err != nil {
  124. log.Error("s.session error (%v)", err)
  125. return
  126. }
  127. } else {
  128. stat := ssn.State
  129. if ssn.State == model.StateReplied {
  130. stat = model.StateRepeated
  131. }
  132. var ip32 uint32
  133. ipv := net.ParseIP(ip)
  134. if ip2 := ipv.To4(); ip2 != nil {
  135. ip32 = model.InetAtoN(ip)
  136. }
  137. ssn = &model.Session{
  138. ID: ssn.ID,
  139. Device: device,
  140. Channel: channel,
  141. IP: ip32,
  142. NetState: netState,
  143. NetOperator: netOperator,
  144. AgencyArea: agencyArea,
  145. Platform: platform,
  146. Browser: browser,
  147. QQ: qq,
  148. Email: email,
  149. State: stat,
  150. LasterTime: cTime,
  151. MTime: cTime,
  152. }
  153. if _, err = s.dao.UpdateSession(c, ssn); err != nil {
  154. log.Error("s.dao.UpdateSession error(%v)", err)
  155. return
  156. }
  157. if content != "" || logURL != "" {
  158. reply = &model.Reply{
  159. SessionID: ssn.ID,
  160. ReplyID: replyID,
  161. Content: content,
  162. ImgURL: "",
  163. LogURL: logURL,
  164. CTime: cTime,
  165. MTime: cTime,
  166. }
  167. if id, err = s.dao.AddReply(c, reply); err != nil {
  168. log.Error("s.dao.AddReply error(%v)", err)
  169. return
  170. }
  171. }
  172. for _, v := range strings.Split(imgURL, ";") {
  173. if v != "" {
  174. cTime = xtime.Time(now.Unix())
  175. reply = &model.Reply{
  176. SessionID: ssn.ID,
  177. ReplyID: replyID,
  178. Content: "",
  179. ImgURL: v,
  180. LogURL: "",
  181. CTime: cTime,
  182. MTime: cTime,
  183. }
  184. if id, err = s.dao.AddReply(c, reply); err != nil {
  185. log.Error("s.dao.AddReply error(%v)", err)
  186. continue
  187. }
  188. }
  189. }
  190. }
  191. }
  192. r = &model.Reply{
  193. ID: id,
  194. ReplyID: replyID,
  195. Type: model.TypeCustomer,
  196. Content: content,
  197. ImgURL: imgURL,
  198. LogURL: logURL,
  199. CTime: cTime,
  200. }
  201. return
  202. }
  203. // AddWebReply add web reply.
  204. func (s *Service) AddWebReply(c context.Context, mid, sid, tagID int64, aid, content, imgURL, netState, netOperator, agencyArea, platform, version, buvid, browser, qq, email string, now time.Time) (r *model.Reply, err error) {
  205. var (
  206. cTime xtime.Time
  207. reply *model.Reply
  208. tx *sql.Tx
  209. rid int
  210. id int64
  211. sessionID int64
  212. replyID string
  213. ip = metadata.String(c, metadata.RemoteIP)
  214. )
  215. if mid != 0 {
  216. replyID = strconv.FormatInt(mid, 10)
  217. }
  218. sessionID = sid
  219. cTime = xtime.Time(now.Unix())
  220. if buvid == "" {
  221. buvid = strconv.FormatInt(now.Unix(), 10)
  222. }
  223. rid, err = s.dao.JudgeSsnRecord(c, sessionID)
  224. if err != nil {
  225. log.Error("s.dao.JudgeSsnRecord error(%v)", err)
  226. return
  227. }
  228. if sessionID > 0 || rid > 0 {
  229. reply = &model.Reply{
  230. SessionID: sessionID,
  231. ReplyID: replyID,
  232. Content: content,
  233. ImgURL: imgURL,
  234. CTime: cTime,
  235. MTime: cTime,
  236. }
  237. tx, err = s.dao.BeginTran(c)
  238. if err != nil {
  239. log.Error("s.dao.Begin error(%v)", err)
  240. return
  241. }
  242. if id, err = s.dao.TxAddReply(tx, reply); err != nil {
  243. log.Error("s.dao.TxAddReply error(%v)", err)
  244. tx.Rollback()
  245. return
  246. }
  247. if err = s.dao.TxUpdateSessionState(tx, model.StateNoReply, sessionID); err != nil {
  248. log.Error("s.dao.TxUpdateSessionState error(%v)", err)
  249. tx.Rollback()
  250. return
  251. }
  252. if err = tx.Commit(); err != nil {
  253. log.Error("tx.Commit(), error(%v)", err)
  254. return
  255. }
  256. r = &model.Reply{
  257. ID: id,
  258. ReplyID: replyID,
  259. Type: model.TypeCustomer,
  260. Content: content,
  261. ImgURL: imgURL,
  262. CTime: cTime,
  263. }
  264. } else {
  265. sessionID, err = s.session(c, mid, tagID, buvid, "", version, aid, content, imgURL, "", "", "", ip, "", netState, netOperator, agencyArea, platform, browser, qq, email, now)
  266. if err == nil {
  267. reply = &model.Reply{
  268. SessionID: sessionID,
  269. ReplyID: replyID,
  270. Content: content,
  271. ImgURL: imgURL,
  272. CTime: cTime,
  273. MTime: cTime,
  274. }
  275. if _, err = s.dao.AddReply(c, reply); err != nil {
  276. log.Error("s.dao.AddReplyNoTx error(%v)", err)
  277. return
  278. }
  279. }
  280. }
  281. return
  282. }
  283. func (s *Service) session(c context.Context, mid, tagID int64, buvid, system, version string, aid, content, imgURL, logURL, device, channel, ip, entrance,
  284. netState, netOperator, agencyArea, platform, browser, qq, email string, now time.Time) (sid int64, err error) {
  285. if platform == "ugc" {
  286. var count int
  287. // 通过平台进行处理
  288. count, err = s.dao.SessionCount(c, mid)
  289. if err != nil {
  290. log.Error("s.dao.SessionCount error(%v)", err)
  291. return
  292. }
  293. if count >= 10 {
  294. err = ecode.FeedbackContentOver
  295. return
  296. }
  297. }
  298. cTime := xtime.Time(now.Unix())
  299. var ip32 uint32
  300. ipv := net.ParseIP(ip)
  301. if ip2 := ipv.To4(); ip2 != nil {
  302. ip32 = model.InetAtoN(ip)
  303. }
  304. ssn := &model.Session{
  305. Buvid: buvid,
  306. System: system,
  307. Version: version,
  308. Mid: mid,
  309. Aid: aid,
  310. Content: content,
  311. ImgURL: imgURL,
  312. LogURL: logURL,
  313. Device: device,
  314. Channel: channel,
  315. IP: ip32,
  316. NetState: netState,
  317. NetOperator: netOperator,
  318. AgencyArea: agencyArea,
  319. Platform: platform,
  320. Browser: browser,
  321. QQ: qq,
  322. Email: email,
  323. State: model.StateNoReply,
  324. LasterTime: cTime,
  325. CTime: cTime,
  326. MTime: cTime,
  327. }
  328. if entrance == "view" || content == "播放器反馈日志" || entrance == _androidLivePink || entrance == _iosLivePink || platform == _androidLivePink || platform == _iosLivePink || model.IsPlayerScreen(tagID) || entrance == _tvYST || platform == _tvYST || entrance == _creativeCenter || platform == _creativeCenter {
  329. ssn.State = model.StateOther
  330. }
  331. var tx *sql.Tx
  332. tx, err = s.dao.BeginTran(c)
  333. if err != nil {
  334. log.Error("s.dao.Begin error(%v)", err)
  335. return
  336. }
  337. sid, err = s.dao.TxAddSession(tx, ssn)
  338. if err != nil {
  339. log.Error("s.dao.TxAddSession error(%v)", err)
  340. tx.Rollback()
  341. return
  342. }
  343. if err = s.dao.TxUpSsnMtime(tx, now, sid); err != nil {
  344. log.Error("s.dao.TxUpSsnMtime error(%v)", err)
  345. tx.Rollback()
  346. return
  347. }
  348. ssn.ID = sid
  349. if tagID > 0 {
  350. if _, err = s.dao.TxAddSessionTag(tx, ssn.ID, tagID, now); err != nil {
  351. log.Error("s.dao.TxAddSessionTag error(%v)", err)
  352. tx.Rollback()
  353. return
  354. }
  355. }
  356. if err = tx.Commit(); err != nil {
  357. log.Error("tx.Commit(), error(%v)", err)
  358. }
  359. return
  360. }
  361. // Replys show all feedback replays
  362. func (s *Service) Replys(c context.Context, buvid, platform, mobiApp, device, system, version, entrance string, mid int64, pn, ps int) (rs []model.Reply, isEndReply bool, err error) {
  363. var (
  364. replyID = buvid
  365. ssns []*model.Session
  366. ssn *model.Session
  367. r model.Reply
  368. tmp []model.Reply
  369. offset int
  370. limit int
  371. start int
  372. end int
  373. rsl int
  374. )
  375. offset = (pn - 1) * ps
  376. limit = ps
  377. if mid != 0 {
  378. if platform == "android" {
  379. if entrance == _androidCreative {
  380. platform = _androidCreative
  381. } else if entrance == _androidLivePink {
  382. platform = _androidLivePink
  383. } else if model.Plat(mobiApp, device) == model.PlatAndroid {
  384. platform = _androidFeedback
  385. } else if model.Plat(mobiApp, device) == model.PlatAndroidI {
  386. platform = _androidFeedbackI
  387. }
  388. } else if platform == "ios" {
  389. if entrance == _iosCreative {
  390. platform = _iosCreative
  391. } else if entrance == _iosLivePink {
  392. platform = _iosLivePink
  393. } else if model.Plat(mobiApp, device) == model.PlatIPhone {
  394. platform = _iosFeedback
  395. } else if model.Plat(mobiApp, device) == model.PlatIPhoneI {
  396. platform = _iosFeedbackI
  397. }
  398. }
  399. if ssns, err = s.dao.SessionByMid(c, mid, platform); err != nil {
  400. log.Error("s.dao.SessionByMid(%d) error(%v)", mid, err)
  401. return
  402. }
  403. if len(ssns) == 0 {
  404. err = ecode.FeedbackSsnNotExist
  405. return
  406. }
  407. replyID = strconv.FormatInt(mid, 10)
  408. if tmp, err = s.dao.ReplysByMid(c, mid, offset, limit); err != nil {
  409. log.Error("s.dao.ReplysByMid(%d) error(%v)", mid, err)
  410. return
  411. }
  412. if len(tmp) > 0 {
  413. rs = tmp
  414. }
  415. for _, ssn = range ssns {
  416. r = model.Reply{
  417. ID: ssn.ID,
  418. ReplyID: replyID,
  419. Type: model.TypeCustomer,
  420. Content: ssn.Content,
  421. ImgURL: ssn.ImgURL,
  422. LogURL: ssn.LogURL,
  423. CTime: ssn.CTime,
  424. }
  425. rs = append(rs, r)
  426. }
  427. sort.Sort(model.Replys(rs))
  428. rsl = len(rs)
  429. end = rsl
  430. if limit > rsl {
  431. start = 0
  432. isEndReply = true
  433. } else {
  434. start = end - limit
  435. }
  436. rs = rs[start:end]
  437. } else {
  438. if ssn, err = s.dao.Session(c, buvid, system, version, mid); err != nil {
  439. log.Error("s.dao.Session(%s,%s,%s,%d) error(%v)", buvid, system, version, mid, err)
  440. return
  441. }
  442. if ssn == nil {
  443. err = ecode.FeedbackSsnNotExist
  444. return
  445. }
  446. if tmp, err = s.dao.Replys(c, ssn.ID, offset, limit); err != nil {
  447. log.Error("s.dao.Replays(%d) error(%v)", ssn.ID, err)
  448. return
  449. }
  450. if len(tmp) == limit {
  451. rs = tmp
  452. } else {
  453. r = model.Reply{
  454. ID: ssn.ID,
  455. ReplyID: replyID,
  456. Type: model.TypeCustomer,
  457. Content: ssn.Content,
  458. ImgURL: ssn.ImgURL,
  459. LogURL: ssn.LogURL,
  460. CTime: ssn.CTime,
  461. }
  462. rs = append(tmp, r)
  463. isEndReply = true
  464. }
  465. sort.Sort(model.Replys(rs))
  466. }
  467. return
  468. }
  469. // Sessions sessions.
  470. func (s *Service) Sessions(c context.Context, mid int64, state string, tagID, platform string, start, end time.Time, ps, pn int) (total int, wssns []*model.WebSession, err error) {
  471. var (
  472. sids, sidsTmp, tids, intersect, sidTmp, sidCut []int64
  473. limit, st, en, sls int
  474. ssnMap map[int64]*model.Session
  475. ssns []*model.Session
  476. )
  477. if mid > 0 {
  478. ssns, err = s.dao.SessionByMid(c, mid, platform)
  479. if err != nil {
  480. log.Error("s.dao.SessionByMid error(%v)", err)
  481. return
  482. }
  483. for _, v := range ssns {
  484. sids = append(sids, v.ID)
  485. }
  486. if tagID != "" {
  487. tids, err = xstr.SplitInts(tagID)
  488. if err != nil {
  489. log.Error("xstr.SplitInts error(%v)", err)
  490. return
  491. }
  492. if len(tids) > 0 {
  493. sidsTmp, err = s.dao.SessionIDByTagID(c, tids)
  494. if err != nil {
  495. log.Error("s.dao.SessionIDByTagID error(%v)", err)
  496. return
  497. }
  498. }
  499. for _, v := range sids {
  500. if contains(sidsTmp, v) {
  501. intersect = append(intersect, v)
  502. }
  503. }
  504. } else {
  505. intersect = sids
  506. }
  507. }
  508. if len(intersect) > 0 {
  509. if state == "" {
  510. ssns, err = s.dao.SSnBySsnIDAllSate(c, intersect, start, end)
  511. if err != nil {
  512. log.Error("s.dao.sSessionBySsnID error(%v)", err)
  513. return
  514. }
  515. } else {
  516. ssns, err = s.dao.SessionBySsnID(c, intersect, state, start, end)
  517. if err != nil {
  518. log.Error("s.dao.sSessionBySsnID error(%v)", err)
  519. return
  520. }
  521. }
  522. ssnMap = make(map[int64]*model.Session, len(ssns))
  523. for _, v := range ssns {
  524. sidTmp = append(sidTmp, v.ID)
  525. ssnMap[v.ID] = v
  526. }
  527. sls = len(sidTmp)
  528. total = sls
  529. limit = ps
  530. if limit > sls {
  531. st = 0
  532. en = sls
  533. } else {
  534. st = (pn - 1) * limit
  535. en = pn * limit
  536. if en > sls {
  537. en = st + (sls % limit)
  538. }
  539. }
  540. }
  541. sidCut = sidTmp[st:en]
  542. if len(sidCut) > 0 {
  543. var tagsMap map[int64][]*model.Tag
  544. tagsMap, err = s.dao.TagBySsnID(c, sidCut)
  545. if err != nil {
  546. log.Error("s.dao.TagBySsnID error(%v)", err)
  547. return
  548. }
  549. for _, v := range sidCut {
  550. wssn := &model.WebSession{}
  551. wssn.Session = ssnMap[v]
  552. tags := tagsMap[v]
  553. if len(tags) == 0 {
  554. wssn.Tag = defaultTag
  555. } else {
  556. wssn.Tag = tags[len(tags)-1]
  557. }
  558. wssns = append(wssns, wssn)
  559. }
  560. }
  561. return
  562. }
  563. // UpdateSessionState up session state.
  564. func (s *Service) UpdateSessionState(c context.Context, state int, sid int64) (err error) {
  565. if err = s.dao.UpdateSessionState(c, state, sid); err != nil {
  566. log.Error("s.dao.UpdateSessionState error(%v)", err)
  567. }
  568. return
  569. }
  570. // Tags tags.
  571. func (s *Service) Tags(c context.Context, mid int64, mold int, platform string) (tag *model.UGCTag, err error) {
  572. tags, err := s.dao.Tags(c, mold, platform)
  573. if err != nil {
  574. log.Error("s.dao.Tags error(%v)", err)
  575. return
  576. }
  577. cnt, err := s.dao.SessionCount(c, mid)
  578. if err != nil {
  579. log.Error("s.dao.SessionCount error(%v)", err)
  580. return
  581. }
  582. tag = &model.UGCTag{
  583. Tags: tags,
  584. Limit: 10 - cnt,
  585. }
  586. return
  587. }
  588. // WebReplys web replys.
  589. func (s *Service) WebReplys(c context.Context, sid, mid int64) (mp []*model.Reply, err error) {
  590. if mp, err = s.dao.WebReplys(c, sid, mid); err != nil {
  591. log.Error("s.dao.WebReplys(%d, %d) error(%v)", sid, mid, err)
  592. return
  593. }
  594. if len(mp) == 0 {
  595. err = ecode.NothingFound
  596. }
  597. return
  598. }
  599. func contains(s []int64, e int64) bool {
  600. for _, a := range s {
  601. if a == e {
  602. return true
  603. }
  604. }
  605. return false
  606. }
  607. // PlayerCheck check for player.
  608. func (s *Service) PlayerCheck(c context.Context, platform, ipChangeTimes int, mid, checkTime, aid, connectSpeed, ioSpeed int64, region, school, cdnip string) (err error) {
  609. var (
  610. ipinfo *locmdl.Info
  611. isp int
  612. ipAddr = metadata.String(c, metadata.RemoteIP)
  613. )
  614. if ipinfo, err = s.locationRPC.Info(c, &locmdl.ArgIP{IP: ipAddr}); err != nil {
  615. log.Error("s.locationRPC.Info(%s) error(%v)", ipAddr, err)
  616. }
  617. if ipinfo != nil {
  618. if strings.Contains(ipinfo.ISP, "移动") {
  619. isp = 1
  620. } else if strings.Contains(ipinfo.ISP, "联通") {
  621. isp = 2
  622. } else if strings.Contains(ipinfo.ISP, "电信") {
  623. isp = 3
  624. } else {
  625. isp = 0
  626. }
  627. }
  628. if _, err = s.dao.InPlayCheck(c, platform, isp, ipChangeTimes, mid, checkTime, aid, connectSpeed, ioSpeed, region, school, ipAddr, cdnip); err != nil {
  629. log.Error("s.dao.InPlayCheck(%d, %d, %d, %d, %d, %d, %d, %d, %s, %s, %s, %s) error(%v)", platform, isp, ipChangeTimes, mid, checkTime, aid, connectSpeed, ioSpeed, region, school, ipAddr, cdnip, err)
  630. }
  631. return
  632. }