You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

165 lines
5.0 KiB

5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
  1. <?php
  2. namespace App\Controller;
  3. use App\Constants\ErrorCode;
  4. use App\Model\Order;
  5. use App\Model\OrderGoods;
  6. use App\Model\OrderMain;
  7. use App\Request\WxminiPayRequest;
  8. use Hyperf\Di\Annotation\Inject;
  9. use App\Service\PurchaseLimitServiceInterface;
  10. use EasyWeChat\Factory;
  11. use Hyperf\DbConnection\Db;
  12. use Hyperf\Guzzle\CoroutineHandler;
  13. use App\Constants\LogLabel;
  14. use App\Commons\Log;
  15. class PaymentController extends BaseController
  16. {
  17. /**
  18. * @Inject
  19. * @var PurchaseLimitServiceInterface
  20. */
  21. protected $purchaseLimitService;
  22. public function wxminiPayOnline(WxminiPayRequest $request){
  23. $data = $request->validated();
  24. $config = config('wxpay');
  25. $app = Factory::payment($config);
  26. $app['guzzle_handler'] = CoroutineHandler::class;
  27. // 待支付的,未超时(15min,900sec)的订单
  28. $orderMain = OrderMain::query()
  29. ->where(['state' => OrderMain::ORDER_STATE_UNPAY, 'id' => $data['order_id']])
  30. ->where('time', '>=', date('Y-m-d H:i:s', (time()-900)))
  31. ->first();
  32. if (empty($orderMain)) {
  33. return $this->result(ErrorCode::PAY_FAILURE, $data,'订单不存在或已失效');
  34. }
  35. //查询订单商品信息
  36. $order = Order::query()
  37. ->where('order_main_id',$data['order_id'])
  38. ->select('id')
  39. ->get()
  40. ->toArray();
  41. $orderGoods = OrderGoods::query()
  42. ->whereIn('order_id',$order)
  43. ->get();
  44. //判断是否有购买多个特价商品
  45. $result = $this->purchaseLimitService->PurchaseLimit($orderGoods);
  46. if(!$result){
  47. $this->log->event(LogLabel::ORDER_LOG, ['msg'=> '支付失败,同一订单不能存在多个特价商品','global_order_id' => $orderMain->global_order_id]);
  48. return $this->result(ErrorCode::PAY_FAILURE, $data,'同一订单不能存在多个特价商品');
  49. }
  50. $payMoney = bcmul(floatval($orderMain->money), 100, 0);
  51. if (env('APP_ENV') != 'prod') {
  52. $payMoney = 1;
  53. }
  54. $result = $app->order->unify([
  55. 'body' => '懒族生活 - 外卖下单',
  56. 'out_trade_no' => $orderMain->global_order_id,
  57. 'total_fee' => $payMoney,
  58. 'notify_url' => config('site_host') . '/wechat/notify/wxminionline',
  59. 'trade_type' => 'JSAPI',
  60. 'openid' => $data['openid'],
  61. ]);
  62. // 返回支付参数给前端
  63. $parameters = [
  64. 'appId' => $result['appid'],
  65. 'timeStamp' => '' . time() . '',
  66. 'nonceStr' => uniqid(),
  67. 'package' => 'prepay_id=' . $result['prepay_id'],
  68. 'signType' => 'MD5'
  69. ];
  70. $parameters['paySign'] = $this->signture($parameters, $config['key']);
  71. return $this->success($parameters);
  72. }
  73. public function wxminiPayOffline(WxminiPayRequest $request){
  74. $data = $request->validated();
  75. $config = config('wxpay');
  76. $app = Factory::payment($config);
  77. $app['guzzle_handler'] = CoroutineHandler::class;
  78. // 待支付的,未超时(15min,900sec)的订单
  79. $orderMain = OrderMain::query()
  80. ->where(['dm_state' => OrderMain::ORDER_STATE_UNPAY, 'id' => $data['order_id']])
  81. ->first();
  82. if (empty($orderMain)) {
  83. return $this->result(ErrorCode::PAY_FAILURE, $data,'订单不存在或已失效');
  84. }
  85. $result = $app->order->unify([
  86. 'body' => '懒族生活 - 当面支付',
  87. 'out_trade_no' => $orderMain->global_order_id,
  88. 'total_fee' => bcmul(floatval($orderMain->money), 100, 0),
  89. 'notify_url' => config('site_host') . '/wechat/notify/wxminioffline',
  90. 'trade_type' => 'JSAPI',
  91. 'openid' => $data['openid'],
  92. ]);
  93. // 返回支付参数给前端
  94. $parameters = [
  95. 'appId' => $result['appid'],
  96. 'timeStamp' => '' . time() . '',
  97. 'nonceStr' => uniqid(),
  98. 'package' => 'prepay_id=' . $result['prepay_id'],
  99. 'signType' => 'MD5'
  100. ];
  101. $parameters['paySign'] = $this->signture($parameters, $config['key']);
  102. return $this->success($parameters);
  103. }
  104. /**
  105. * 支付参数加签
  106. * @param $parameters
  107. * @param $key
  108. * @return string
  109. */
  110. private function signture($parameters, $key)
  111. {
  112. // 按字典序排序参数
  113. ksort($parameters);
  114. // http_query
  115. $queryParams = $this->http_query($parameters);
  116. // 加入KEY
  117. $queryParams = $queryParams . "&key=" . $key;
  118. // MD5加密
  119. $queryParams = md5($queryParams);
  120. // 字符转为大写
  121. return strtoupper($queryParams);
  122. }
  123. /**
  124. * 参数转为http query字串
  125. * @param $parameters
  126. * @return string
  127. */
  128. private function http_query($parameters) {
  129. $http_query = [];
  130. foreach ($parameters as $key => $value) {
  131. $http_query[] = $key.'='.$value;
  132. }
  133. return implode('&', $http_query);
  134. }
  135. }