withdraw.go 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. package service
  2. import (
  3. "context"
  4. "fmt"
  5. "strconv"
  6. "time"
  7. "go-common/app/interface/main/growup/model"
  8. "go-common/library/log"
  9. )
  10. var (
  11. _withdrawing = 1 // 处理中
  12. _withdrawSuccess = 2 // 提现成功
  13. // _withdrawFail = 3 // 提现失败
  14. )
  15. // GetWithdraw get up withdraw
  16. func (s *Service) GetWithdraw(c context.Context, dateVersion string, from, limit int) (count int, withdrawVos []*model.WithdrawVo, err error) {
  17. count, upAccounts, err := s.UpWithdraw(c, dateVersion, from, limit)
  18. if err != nil {
  19. log.Error("s.UpWithdraw error(%v)", err)
  20. return
  21. }
  22. mids := make([]int64, len(upAccounts))
  23. for i, up := range upAccounts {
  24. mids[i] = up.MID
  25. }
  26. withdrawVos = make([]*model.WithdrawVo, 0)
  27. if len(mids) == 0 {
  28. return
  29. }
  30. upIncomeWithdrawMap, err := s.dao.QueryUpWithdrawByMids(c, mids, dateVersion)
  31. if err != nil {
  32. log.Error("s.dao.QueryUpWithdrawByMids error(%v)", err)
  33. return
  34. }
  35. for _, up := range upAccounts {
  36. if upIncomeWithdraw, ok := upIncomeWithdrawMap[up.MID]; ok && upIncomeWithdraw.State == _withdrawing {
  37. vo := &model.WithdrawVo{
  38. MID: up.MID,
  39. ThirdCoin: float64(up.TotalUnwithdrawIncome) * float64(0.01),
  40. ThirdOrderNo: strconv.FormatInt(upIncomeWithdraw.ID, 10),
  41. CTime: time.Unix(int64(upIncomeWithdraw.CTime), 0).Format("2006-01-02 15:04:05"),
  42. NotifyURL: "http://up-profit.bilibili.co/allowance/api/x/internal/growup/up/withdraw/success",
  43. }
  44. withdrawVos = append(withdrawVos, vo)
  45. }
  46. }
  47. return
  48. }
  49. // UpWithdraw get up withdraw
  50. func (s *Service) UpWithdraw(c context.Context, dateVersion string, from, limit int) (count int, upAccounts []*model.UpAccount, err error) {
  51. count, err = s.dao.GetUpAccountCount(c, dateVersion)
  52. if err != nil {
  53. log.Error("s.dao.GetUpAccountCount error(%v)", err)
  54. return
  55. }
  56. if count <= 0 {
  57. return
  58. }
  59. upAccounts, err = s.dao.QueryUpAccountByDate(c, dateVersion, from, limit)
  60. if err != nil {
  61. log.Error("s.dao.QueryUpAccountByDate error(%v)", err)
  62. return
  63. }
  64. if len(upAccounts) == 0 {
  65. return
  66. }
  67. mids := make([]int64, len(upAccounts))
  68. for i, up := range upAccounts {
  69. mids[i] = up.MID
  70. }
  71. // get up_income_withdraw by mids and date
  72. upIncomeWithdrawMap, err := s.dao.QueryUpWithdrawByMids(c, mids, dateVersion)
  73. if err != nil {
  74. log.Error("s.dao.QueryUpWithdrawByMids error(%v)", err)
  75. return
  76. }
  77. for _, up := range upAccounts {
  78. if _, ok := upIncomeWithdrawMap[up.MID]; !ok {
  79. upIncomeWithdraw := &model.UpIncomeWithdraw{
  80. MID: up.MID,
  81. WithdrawIncome: up.TotalUnwithdrawIncome,
  82. DateVersion: dateVersion,
  83. State: _withdrawing,
  84. }
  85. err = s.InsertUpWithdrawRecord(c, upIncomeWithdraw)
  86. if err != nil {
  87. log.Error("s.InsertUpWithdrawRecord error(%v)", err)
  88. return
  89. }
  90. }
  91. }
  92. return
  93. }
  94. // InsertUpWithdrawRecord insert up_withdraw_income record
  95. func (s *Service) InsertUpWithdrawRecord(c context.Context, upIncomeWithdraw *model.UpIncomeWithdraw) (err error) {
  96. result, err := s.dao.InsertUpWithdrawRecord(c, upIncomeWithdraw)
  97. if err != nil {
  98. log.Error("s.dao.InsertUpIncomeWithdraw error(%v)", err)
  99. return
  100. }
  101. if result < 1 {
  102. log.Error("s.dao.InsertUpIncomeWithdraw error mid(%d), dateVersion(%s)", upIncomeWithdraw.MID, upIncomeWithdraw.DateVersion)
  103. return
  104. }
  105. return
  106. }
  107. // WithdrawSuccess withdraw success callback
  108. func (s *Service) WithdrawSuccess(c context.Context, orderNo int64, tradeStatus int) (err error) {
  109. upWithdraw, err := s.dao.QueryUpWithdrawByID(c, orderNo)
  110. if err != nil {
  111. log.Error("s.dao.QueryUpWithdrawByID error(%v)", err)
  112. return
  113. }
  114. if tradeStatus != _withdrawSuccess {
  115. log.Info("param tradeStatus(%d) != withdraw success(2)", tradeStatus)
  116. return
  117. }
  118. if upWithdraw.State == _withdrawSuccess {
  119. log.Info("withdraw has successed already")
  120. return
  121. }
  122. tx, err := s.dao.BeginTran(c)
  123. if err != nil {
  124. log.Error("s.dao.BeginTran error(%v)", err)
  125. return
  126. }
  127. // update up_income_withdraw state
  128. rows, err := s.dao.TxUpdateUpWithdrawState(tx, orderNo, _withdrawSuccess)
  129. if err != nil {
  130. tx.Rollback()
  131. log.Error("s.dao.UpdateUpWithdrawState error(%v)", err)
  132. return
  133. }
  134. if rows != 1 {
  135. tx.Rollback()
  136. log.Error("s.dao.UpdateUpWithdrawState Update withdraw record error id(%d)", orderNo)
  137. return
  138. }
  139. // update up_account withdraw
  140. rows, err = s.dao.TxUpdateUpAccountWithdraw(tx, upWithdraw.MID, upWithdraw.WithdrawIncome)
  141. if err != nil {
  142. tx.Rollback()
  143. log.Error("s.dao.UpdateUpAccountWithdraw error(%v)", err)
  144. return
  145. }
  146. if rows != 1 {
  147. tx.Rollback()
  148. log.Error("s.dao.UpdateUpAccountWithdraw Update up account record error id(%d)", orderNo)
  149. return
  150. }
  151. maxUpWithdrawDateVersion, err := s.dao.TxQueryMaxUpWithdrawDateVersion(tx, upWithdraw.MID)
  152. if err != nil {
  153. tx.Rollback()
  154. log.Error("s.dao.QueryMaxUpWithdrawDateVersion error(%v)", err)
  155. return
  156. }
  157. time := 0
  158. var version int64
  159. for {
  160. version, err = s.dao.TxQueryUpAccountVersion(tx, upWithdraw.MID)
  161. if err != nil {
  162. tx.Rollback()
  163. log.Error("s.dao.QueryUpAccountVersion error(%v)", err)
  164. return
  165. }
  166. if maxUpWithdrawDateVersion == "" {
  167. maxUpWithdrawDateVersion = upWithdraw.DateVersion
  168. }
  169. rows, err = s.dao.TxUpdateUpAccountUnwithdrawIncome(tx, upWithdraw.MID, maxUpWithdrawDateVersion, version)
  170. if err != nil {
  171. tx.Rollback()
  172. log.Error("s.dao.UpdateUpAccountUnwithdrawIncome error(%v)", err)
  173. return
  174. }
  175. if rows == 1 {
  176. if err = tx.Commit(); err != nil {
  177. log.Error("tx.Commit error")
  178. return err
  179. }
  180. break
  181. }
  182. time++
  183. if time >= 10 {
  184. tx.Rollback()
  185. log.Info("try to synchronize unwithdraw income 10 times error mid(%d)", upWithdraw.MID)
  186. err = fmt.Errorf("try to synchronize unwithdraw income 10 times error mid(%d)", upWithdraw.MID)
  187. break
  188. }
  189. }
  190. return
  191. }
  192. // WithdrawDetail get withdraw detail
  193. func (s *Service) WithdrawDetail(c context.Context, mid int64) (upWithdraws []*model.UpIncomeWithdraw, err error) {
  194. return s.dao.QueryUpWithdrawByMID(c, mid)
  195. }