Pay.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. <?php
  2. namespace addons\shopro\controller;
  3. use Psr\Http\Message\ResponseInterface;
  4. use think\exception\HttpResponseException;
  5. use app\admin\model\shopro\Pay as PayModel;
  6. use addons\shopro\service\pay\PayOper;
  7. use addons\shopro\library\pay\PayService;
  8. use app\admin\model\shopro\order\Order;
  9. use app\admin\model\shopro\trade\Order as TradeOrderModel;
  10. use think\Log;
  11. use think\Db;
  12. use addons\shopro\service\pay\PayRefund;
  13. use Yansongda\Pay\Pay as YansongdaPay;
  14. class Pay extends Common
  15. {
  16. protected $noNeedLogin = ['alipay', 'notify', 'notifyRefund'];
  17. protected $noNeedRight = ['*'];
  18. public function prepay()
  19. {
  20. $this->repeatFilter(); // 防止连点
  21. check_env(['yansongda']);
  22. $user = auth_user();
  23. $order_sn = $this->request->post('order_sn');
  24. $payment = $this->request->post('payment');
  25. $openid = $this->request->post('openid', '');
  26. $money = $this->request->post('money', 0);
  27. $money = $money > 0 ? $money : 0;
  28. $platform = $this->request->header('platform');
  29. list($order, $order_type) = $this->getOrderInstance($order_sn);
  30. $order = $order->where('user_id', $user->id)->where('order_sn', $order_sn)->find();
  31. if (!$order) {
  32. $this->error(__('No Results were found'));
  33. }
  34. if (in_array($order->status, [$order::STATUS_CLOSED, $order::STATUS_CANCEL])) {
  35. $this->error('订单已失效');
  36. }
  37. if (in_array($order->status, [$order::STATUS_PAID, $order::STATUS_COMPLETED])) {
  38. $this->error('订单已支付');
  39. }
  40. if ($order_type == 'order' && $order->isOffline($order)) {
  41. // 已经货到付款
  42. $this->error('已下单成功');
  43. }
  44. if (!$payment || !in_array($payment, ['wechat', 'alipay', 'money', 'offline'])) {
  45. $this->error('支付类型不能为空');
  46. }
  47. // pay 实例
  48. $payOper = new PayOper();
  49. if ($money && $order_type == 'order') {
  50. // 余额混合支付
  51. $order = Db::transaction(function () use ($payOper, $order, $order_type, $money) {
  52. // 加锁读订单
  53. $order = $order->lock(true)->find($order->id);
  54. // 余额支付
  55. $order = $payOper->money($order, $money, $order_type);
  56. return $order;
  57. });
  58. if (in_array($order->status, [$order::STATUS_PAID, $order::STATUS_COMPLETED])) {
  59. $this->success('订单支付成功', $order);
  60. }
  61. }
  62. if ($payment == 'money' && $order_type == 'order') {
  63. // 余额支付
  64. $order = Db::transaction(function () use ($payOper, $order, $order_type) {
  65. // 加锁读订单
  66. $order = $order->lock(true)->find($order->id);
  67. $order = $payOper->money($order, $order->remain_pay_fee, $order_type);
  68. return $order;
  69. });
  70. if ($order->status != $order::STATUS_PAID) {
  71. $this->error('订单支付失败');
  72. }
  73. $this->success('订单支付成功', $order);
  74. }
  75. if ($payment == 'offline' && $order_type == 'order') {
  76. if (!isset($order->ext['offline_status']) || $order->ext['offline_status'] != 'enable') {
  77. $this->error('订单不支持货到付款');
  78. }
  79. // 货到付款
  80. $order = Db::transaction(function () use ($payOper, $order, $order_type) {
  81. // 加锁读订单
  82. $order = $order->lock(true)->find($order->id);
  83. $order = $payOper->offline($order, 0, $order_type); // 增加 0 记录
  84. return $order;
  85. });
  86. if ($order->status != $order::STATUS_PAID) {
  87. $this->success('下单成功', $order); // 货到付款
  88. }
  89. $this->success('订单支付成功', $order);
  90. }
  91. // 微信支付宝(第三方)付款
  92. $payModel = $payOper->{$payment}($order, $order->remain_pay_fee, $order_type);
  93. $order_data = [
  94. 'order_id' => $order->id,
  95. 'out_trade_no' => $payModel->pay_sn,
  96. 'total_amount' => $payModel->pay_fee, // 剩余支付金额
  97. ];
  98. // 微信公众号,小程序支付,必须有 openid
  99. if ($payment == 'wechat') {
  100. if (in_array($platform, ['WechatOfficialAccount', 'WechatMiniProgram'])) {
  101. if (isset($openid) && $openid) {
  102. // 如果传的有 openid
  103. $order_data['payer']['openid'] = $openid;
  104. } else {
  105. // 没有 openid 默认拿下单人的 openid
  106. $oauth = \app\admin\model\shopro\ThirdOauth::where([
  107. 'user_id' => $order->user_id,
  108. 'provider' => 'Wechat',
  109. 'platform' => lcfirst(str_replace('Wechat', '', $platform))
  110. ])->find();
  111. $order_data['payer']['openid'] = $oauth ? $oauth->openid : '';
  112. }
  113. if (empty($order_data['payer']['openid'])) {
  114. // 缺少 openid
  115. $this->error('miss_openid', -1);
  116. }
  117. }
  118. $order_data['description'] = '商城订单支付';
  119. } else {
  120. $order_data['subject'] = '商城订单支付';
  121. }
  122. $payService = new PayService($payment);
  123. try {
  124. $result = $payService->pay($order_data);
  125. } catch (\Yansongda\Pay\Exception\Exception $e) {
  126. $this->error('支付失败' . (config('app_debug') ? ":" . $e->getMessage() : ',请重试'));
  127. } catch (HttpResponseException $e) {
  128. $data = $e->getResponse()->getData();
  129. $message = $data ? ($data['msg'] ?? '') : $e->getMessage();
  130. $this->error('支付失败' . (config('app_debug') ? ":" . $message : ',请重试'));
  131. }
  132. if ($platform == 'App') {
  133. if ($payment == 'wechat') {
  134. // Yansongda\Supports\Collection,可当数组,可当字符串,这里不用处理
  135. } else {
  136. // Guzzle
  137. $result = $result->getBody()->getContents();
  138. }
  139. }
  140. $this->success('', [
  141. 'pay_data' => $result,
  142. ]);
  143. }
  144. /**
  145. * 支付宝网页支付
  146. */
  147. public function alipay()
  148. {
  149. $pay_sn = $this->request->get('pay_sn');
  150. $platform = $this->request->get('platform');
  151. $payModel = PayModel::where('pay_sn', $pay_sn)->find();
  152. if (!$payModel || $payModel->status != PayModel::PAY_STATUS_UNPAID) {
  153. $this->error("支付单不存在或已支付");
  154. }
  155. try {
  156. $order_data = [
  157. 'order_id' => $payModel->order_id,
  158. 'out_trade_no' => $payModel->pay_sn,
  159. 'total_amount' => $payModel->pay_fee,
  160. 'subject' => '商城订单支付',
  161. ];
  162. $payService = new PayService('alipay', $platform);
  163. $result = $payService->pay($order_data, [], 'url');
  164. $result = $result->getBody()->getContents();
  165. echo $result;
  166. } catch (\Exception $e) {
  167. echo $e->getMessage();
  168. } catch (HttpResponseException $e) {
  169. $data = $e->getResponse()->getData();
  170. $message = $data ? ($data['msg'] ?? '') : $e->getMessage();
  171. echo '支付失败' . (config('app_debug') ? ":" . $message : ',请重试');
  172. }
  173. }
  174. /**
  175. * 支付成功回调
  176. */
  177. public function notify()
  178. {
  179. Log::write('pay-notify-comein:');
  180. $payment = $this->request->param('payment');
  181. $platform = $this->request->param('platform');
  182. $payService = new PayService($payment, $platform);
  183. $result = $payService->notify(function ($data, $originData = []) use ($payment) {
  184. Log::write('pay-notify-data:' . json_encode($data));
  185. $out_trade_no = $data['out_trade_no'];
  186. // 查询 pay 交易记录
  187. $payModel = PayModel::where('pay_sn', $out_trade_no)->find();
  188. if (!$payModel || $payModel->status != PayModel::PAY_STATUS_UNPAID) {
  189. // 订单不存在,或者订单已支付
  190. return YansongdaPay::$payment()->success();
  191. }
  192. Db::transaction(function () use ($payModel, $data, $originData, $payment) {
  193. $notify = [
  194. 'pay_sn' => $data['out_trade_no'],
  195. 'transaction_id' => $data['transaction_id'],
  196. 'notify_time' => $data['notify_time'],
  197. 'buyer_info' => $data['buyer_info'],
  198. 'payment_json' => $originData ? json_encode($originData) : json_encode($data),
  199. 'pay_fee' => $data['pay_fee'], // 微信的已经*100处理过了
  200. 'pay_type' => $payment // 支付方式
  201. ];
  202. // pay 实例
  203. $payOper = new PayOper($payModel->user_id);
  204. $payOper->notify($payModel, $notify);
  205. });
  206. return YansongdaPay::$payment()->success();
  207. });
  208. return $this->payResponse($result, $payment);
  209. }
  210. /**
  211. * 微信退款回调 (仅微信用,支付宝走支付回调 notify 方法)
  212. *
  213. * @return void
  214. */
  215. public function notifyRefund()
  216. {
  217. Log::write('pay-notify-refund-comein:');
  218. $payment = $this->request->param('payment');
  219. $platform = $this->request->param('platform');
  220. $payService = new PayService($payment, $platform);
  221. $result = $payService->notifyRefund(function ($data, $originData) use ($payment, $platform) {
  222. Log::write('pay-notify-refund-result:' . json_encode($data));
  223. Db::transaction(function () use ($data, $originData, $payment) {
  224. $out_refund_no = $data['out_refund_no'];
  225. $out_trade_no = $data['out_trade_no'];
  226. // 交给退款实例处理
  227. $refund = new PayRefund();
  228. $refund->notify([
  229. 'out_trade_no' => $out_trade_no,
  230. 'out_refund_no' => $out_refund_no,
  231. 'payment_json' => json_encode($originData),
  232. ]);
  233. });
  234. return YansongdaPay::$payment()->success();
  235. });
  236. return $this->payResponse($result, $payment);
  237. }
  238. /**
  239. * 处理返回结果 tp5 不能直接 return YansongdaPay::$payment()->success()
  240. *
  241. * @param object|string $result
  242. * @param string|null $payment
  243. * @return void
  244. */
  245. private function payResponse($result = null, $payment = null)
  246. {
  247. if ($result instanceof ResponseInterface) {
  248. $content = $result->getBody()->getContents();
  249. $content = $payment == 'wechat' ? json_decode($content, true) : $content;
  250. return response($content, 200, [], ($payment == 'wechat' ? 'json' : ''));
  251. }
  252. return $result;
  253. }
  254. /**
  255. * 根据订单号获取订单实例
  256. *
  257. * @param [type] $order_sn
  258. * @return void
  259. */
  260. private function getOrderInstance($order_sn)
  261. {
  262. if (strpos($order_sn, 'TO') === 0) {
  263. // 交易订单
  264. $order_type = 'trade_order';
  265. $order = new TradeOrderModel();
  266. } else {
  267. // 订单
  268. $order_type = 'order';
  269. $order = new Order();
  270. }
  271. return [$order, $order_type];
  272. }
  273. }