UserBalancesController.ts 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. import Logger from '@ioc:Adonis/Core/Logger'
  2. import type { HttpContextContract } from '@ioc:Adonis/Core/HttpContext'
  3. import UserBalance from 'App/Models/UserBalance'
  4. import PaginationService from 'App/Services/PaginationService'
  5. import { schema } from '@ioc:Adonis/Core/Validator'
  6. import UserBalanceService from 'App/Services/UserBalanceService'
  7. import Decimal from 'decimal.js'
  8. import { BalanceRecordType } from 'App/Models/BalanceRecord'
  9. import qs from 'qs'
  10. import { format } from 'date-fns'
  11. import { createHash } from 'crypto'
  12. import axios from 'axios'
  13. import { InternalServerException } from 'App/Exceptions/Common'
  14. import { Exception } from '@adonisjs/core/build/standalone'
  15. import RechargeOrder, { RechargeOrderStatus } from 'App/Models/RechargeOrder'
  16. import { DateTime } from 'luxon'
  17. export default class UserBalancesController {
  18. private paginationService = new PaginationService(UserBalance)
  19. public async index({ request }: HttpContextContract) {
  20. return await this.paginationService.paginate(request.all())
  21. }
  22. public async store({ request }: HttpContextContract) {
  23. const data: any = await request.validate({
  24. schema: schema.create({
  25. userId: schema.number(),
  26. balance: schema.string()
  27. })
  28. })
  29. const userBalance = await UserBalance.create(data)
  30. return userBalance
  31. }
  32. public async show({ params, bouncer }: HttpContextContract) {
  33. const row = await UserBalanceService.getBalance(params.id)
  34. await bouncer.authorize('owner', row)
  35. return row
  36. }
  37. public async update({ params, request }: HttpContextContract) {
  38. const userBalance = await UserBalance.findOrFail(params.id)
  39. const data: any = await request.validate({
  40. schema: schema.create({
  41. userId: schema.number.optional(),
  42. amount: schema.number.optional(),
  43. type: schema.string.optional(),
  44. description: schema.string.optional()
  45. })
  46. })
  47. userBalance.merge(data)
  48. await userBalance.save()
  49. return userBalance
  50. }
  51. public async destroy({ params }: HttpContextContract) {
  52. const userBalance = await UserBalance.findOrFail(params.id)
  53. await userBalance.delete()
  54. }
  55. // public async recharge({ request }: HttpContextContract) {
  56. // console.log(JSON.stringify(request.all(), null, 4))
  57. // UserBalanceService.modifiyBalance({
  58. // userId: parseInt(request.param('userId')),
  59. // amount: new Decimal(request.input('id')),
  60. // type: BalanceRecordType.Recharge
  61. // })
  62. // const data = request.all()
  63. // return {
  64. // ok: true,
  65. // data: {
  66. // id: data.id,
  67. // latest_receipt: true,
  68. // transaction: data.transaction
  69. // }
  70. // }
  71. // }
  72. public async recharge({ request, auth }: HttpContextContract) {
  73. const amount = '100'
  74. const order = await RechargeOrder.create({
  75. userId: auth.user!.id,
  76. amount: new Decimal(amount),
  77. status: RechargeOrderStatus.Pending
  78. })
  79. const head = {
  80. mchtId: '2000713000197336',
  81. version: '20',
  82. biz: 'qr101'
  83. }
  84. const body = {
  85. orderId: order.id.toString(),
  86. orderTime: format(new Date(), 'yyyyMMddHHmmss'),
  87. amount: amount,
  88. currencyType: 'IDR',
  89. goods: 'recharge',
  90. notifyUrl: 'https://kaliartdrama.run.place/api/userBalances/rechargeNotify',
  91. callBackUrl: 'https://www.baidu.com',
  92. email: 'abc@example.com',
  93. phone: '+006281234321234',
  94. name: 'zhangsan'
  95. }
  96. const str = qs.stringify(body, {
  97. encode: false,
  98. sort: (a, b) => {
  99. return a.localeCompare(b)
  100. }
  101. })
  102. Logger.info('param string=%s', str)
  103. const md5 = createHash('md5')
  104. const sign = md5.update(str + '&key=5cde8dfe0c894429bb6b86ec05f8406f').digest('hex')
  105. Logger.info('sign=%s', sign)
  106. const res = await axios.post('https://pre.pay.haodamall.com/gateway/api/commPay', {
  107. head,
  108. body,
  109. sign
  110. })
  111. Logger.info('res=%s', JSON.stringify(res.data, null, 4))
  112. if (res.data.head.respCode === '0000') {
  113. return res.data.body
  114. } else {
  115. throw new Exception(res.data.head.respMsg)
  116. }
  117. }
  118. public async rechargeNotify({ request }: HttpContextContract) {
  119. const data = request.all()
  120. Logger.info('rechargeNotify', data)
  121. if (data.head.respCode === '0000') {
  122. const order = await RechargeOrder.findOrFail(parseInt(data.body.orderId))
  123. if (order.status === RechargeOrderStatus.Pending) {
  124. order.status = RechargeOrderStatus.Success
  125. order.transactionId = data.body.tradeId
  126. order.payTime = DateTime.now()
  127. await order.save()
  128. await UserBalanceService.modifiyBalance({
  129. userId: order.userId,
  130. amount: order.amount,
  131. type: BalanceRecordType.Recharge
  132. })
  133. }
  134. }
  135. return 'SUCCESS'
  136. }
  137. }