BitpayX.php 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. <?php
  2. namespace App\Http\Controllers\Gateway;
  3. use App\Models\Payment;
  4. use Auth;
  5. use Http;
  6. use Illuminate\Http\JsonResponse;
  7. use Log;
  8. use Response;
  9. class BitpayX extends AbstractPayment
  10. {
  11. public function purchase($request): JsonResponse
  12. {
  13. $payment = $this->creatNewPayment(Auth::id(), $request->input('id'), $request->input('amount'));
  14. $data = [
  15. 'merchant_order_id' => $payment->trade_no,
  16. 'price_amount' => $payment->amount,
  17. 'price_currency' => 'CNY',
  18. 'title' => '支付单号:'.$payment->trade_no,
  19. 'description' => sysConfig('subject_name') ?: sysConfig('website_name'),
  20. 'callback_url' => route('payment.notify', ['method' => 'bitpayx']),
  21. 'success_url' => route('invoice'),
  22. 'cancel_url' => route('invoice'),
  23. 'token' => $this->sign($payment->trade_no),
  24. ];
  25. $result = $this->sendRequest($data);
  26. if ($result['status'] === 200 || $result['status'] === 201) {
  27. $result['payment_url'] .= '&lang=zh';
  28. $payment->update(['url' => $result['payment_url']]);
  29. return Response::json(['status' => 'success', 'url' => $result['payment_url'], 'message' => '创建订单成功!']);
  30. }
  31. Log::warning('创建订单错误:'.var_export($result, true));
  32. return Response::json(['status' => 'fail', 'message' => '创建订单失败!'.$result['error']]);
  33. }
  34. private function sign($tradeNo): string
  35. {
  36. $data = [
  37. 'merchant_order_id' => $tradeNo,
  38. 'secret' => sysConfig('bitpay_secret'),
  39. 'type' => 'FIAT',
  40. ];
  41. return $this->aliStyleSign($data, sysConfig('bitpay_secret'));
  42. }
  43. private function sendRequest($data, $type = 'createOrder')
  44. {
  45. $client = Http::baseUrl('https://api.mugglepay.com/v1/')
  46. ->timeout(15)
  47. ->withHeaders([
  48. 'token' => sysConfig('bitpay_secret'),
  49. 'content-type' => 'application/json',
  50. ]);
  51. if ($type === 'query') {
  52. $response = $client->get('orders/merchant_order_id/status?id='.$data['merchant_order_id']);
  53. } else {// Create Order
  54. $response = $client->post('orders', ['body' => json_encode($data)]);
  55. }
  56. if ($response->failed()) {
  57. Log::error('BitPayX请求支付错误:'.var_export($response, true));
  58. }
  59. return $response->json();
  60. }
  61. //Todo: Postman虚拟测试通过,需要真实数据参考验证
  62. public function notify($request): void
  63. {
  64. $tradeNo = $request->input(['merchant_order_id']);
  65. if ($request->input(['status']) === 'PAID' && hash_equals($this->sign($tradeNo), $request->input(['token']))) {
  66. $payment = Payment::whereTradeNo($tradeNo)->first();
  67. if ($payment) {
  68. $ret = $payment->order->update(['status' => 2]);
  69. if ($ret) {
  70. exit(json_encode(['status' => 200]));
  71. }
  72. }
  73. }
  74. exit(json_encode(['status' => 400]));
  75. }
  76. }