Pay.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. <?php
  2. namespace addons\shopro\controller;
  3. use addons\epay\library\Service;
  4. use fast\Random;
  5. use think\addons\Controller;
  6. use addons\shopro\exception\Exception;
  7. use addons\shopro\model\Order;
  8. use addons\shopro\model\User;
  9. use think\Db;
  10. use think\Log;
  11. class Pay extends Base
  12. {
  13. protected $noNeedLogin = ['prepay', 'notifyx', 'notifyr', 'alipay'];
  14. protected $noNeedRight = ['*'];
  15. /**
  16. * 支付宝网页支付
  17. */
  18. public function alipay()
  19. {
  20. $order_sn = $this->request->get('order_sn');
  21. $order = Order::where('order_sn', $order_sn)->find();
  22. try {
  23. if (!$order) {
  24. throw new \Exception("订单不存在");
  25. }
  26. if ($order->status > 0) {
  27. throw new \Exception("订单已支付");
  28. }
  29. if ($order->status < 0) {
  30. throw new \Exception("订单已失效");
  31. }
  32. $notify_url = $this->request->root(true) . '/addons/shopro/pay/notifyx/payment/alipay/platform/H5';
  33. $pay = new \addons\shopro\library\PayService('alipay', 'url', $notify_url);
  34. $order_data = [
  35. 'out_trade_no' => $order->order_sn,
  36. 'total_fee' => $order->total_fee,
  37. 'subject' => '商城订单支付',
  38. ];
  39. $result = $pay->create($order_data);
  40. $result = $result->getContent();
  41. echo $result;
  42. } catch (\Exception $e) {
  43. echo $e->getMessage();
  44. }
  45. // $this->assign('result', $result);
  46. // return $this->view->fetch();
  47. }
  48. /**
  49. * 拉起支付
  50. */
  51. public function prepay()
  52. {
  53. $order_sn = $this->request->post('order_sn');
  54. $payment = $this->request->post('payment');
  55. $openid = $this->request->post('openid', '');
  56. $platform = request()->header('platform');
  57. $order = Order::nopay()->where('order_sn', $order_sn)->find();
  58. if (!$order) {
  59. $this->error("订单不存在");
  60. }
  61. if (!$payment || !in_array($payment, ['wechat', 'alipay', 'wallet'])) {
  62. $this->error("支付类型不能为空");
  63. }
  64. if ($payment == 'wallet') {
  65. // 余额支付
  66. $this->walletPay($order, $payment, $platform);
  67. }
  68. $order_data = [
  69. 'out_trade_no' => $order->order_sn,
  70. 'total_fee' => $order->total_fee,
  71. ];
  72. // 微信公众号,小程序支付,必须有 openid
  73. if ($payment == 'wechat') {
  74. if (in_array($platform, ['wxOfficialAccount', 'wxMiniProgram'])) {
  75. if (isset($openid) && $openid) {
  76. // 如果传的有 openid
  77. $order_data['openid'] = $openid;
  78. } else {
  79. // 没有 openid 默认拿下单人的 openid
  80. $oauth = \addons\shopro\model\UserOauth::where([
  81. 'user_id' => $order->user_id,
  82. 'provider' => 'Wechat',
  83. 'platform' => $platform
  84. ])->find();
  85. $order_data['openid'] = $oauth ? $oauth->openid : '';
  86. }
  87. if (empty($order_data['openid'])) {
  88. // 缺少 openid
  89. return $this->success('缺少 openid', 'no_openid');
  90. }
  91. }
  92. $order_data['body'] = '商城订单支付';
  93. } else {
  94. $order_data['subject'] = '商城订单支付';
  95. }
  96. $notify_url = $this->request->root(true) . '/addons/shopro/pay/notifyx/payment/' . $payment . '/platform/' . $platform;
  97. $pay = new \addons\shopro\library\PayService($payment, $platform, $notify_url);
  98. $result = $pay->create($order_data);
  99. if ($platform == 'App') {
  100. $result = $result->getContent();
  101. }
  102. if ($platform == 'H5' && $payment == 'wechat') {
  103. $result = $result->getContent();
  104. }
  105. return $this->success('获取预付款成功', [
  106. 'pay_data' => $result,
  107. 'pay_action' => $pay->method,
  108. ]);
  109. }
  110. // 余额支付
  111. public function walletPay ($order, $type, $method) {
  112. $order = Db::transaction(function () use ($order, $type, $method) {
  113. // 重新加锁读,防止连点问题
  114. $order = Order::nopay()->where('order_sn', $order->order_sn)->lock(true)->find();
  115. if (!$order) {
  116. $this->error("订单已支付");
  117. }
  118. $total_fee = $order->total_fee;
  119. // 扣除余额
  120. $user = User::info();
  121. if (is_null($user)) {
  122. // 没有登录,请登录
  123. $this->error(__('Please login first'), null, 401);
  124. }
  125. User::money(-$total_fee, $user->id, 'wallet_pay', $order->id, '',[
  126. 'order_id' => $order->id,
  127. 'order_sn' => $order->order_sn,
  128. ]);
  129. // 支付后流程
  130. $notify = [
  131. 'order_sn' => $order['order_sn'],
  132. 'transaction_id' => '',
  133. 'notify_time' => date('Y-m-d H:i:s'),
  134. 'buyer_email' => $user->id,
  135. 'pay_fee' => $order->total_fee,
  136. 'pay_type' => 'wallet' // 支付方式
  137. ];
  138. $notify['payment_json'] = json_encode($notify);
  139. $order->paymentProcess($order, $notify);
  140. return $order;
  141. });
  142. $this->success('支付成功', $order);
  143. }
  144. /**
  145. * 支付成功回调
  146. */
  147. public function notifyx()
  148. {
  149. Log::write('notifyx-comein:');
  150. $payment = $this->request->param('payment');
  151. $platform = $this->request->param('platform');
  152. $pay = new \addons\shopro\library\PayService($payment, $platform);
  153. $result = $pay->notify(function ($data, $pay) use ($payment, $platform) {
  154. Log::write('notifyx-result:'. json_encode($data));
  155. // { // 微信回调参数
  156. // "appid":"wx39cd0799d4567dd0",
  157. // "bank_type":"OTHERS",
  158. // "cash_fee":"1",
  159. // "fee_type":"CNY",
  160. // "is_subscribe":"N",
  161. // "mch_id":"1481069012",
  162. // "nonce_str":"dPpcZ6AzCDU8daNC",
  163. // "openid":"oD9ko4x7QMDQPZEuN8V5jtZjie3g",
  164. // "out_trade_no":"202010240834057843002700",
  165. // "result_code":"SUCCESS",
  166. // "return_code":"SUCCESS",
  167. // "sign":"3103B6D06F13D2B3959C5B3F7F1FD61C",
  168. // "time_end":"20200407102424",
  169. // "total_fee":"1",
  170. // "trade_type":"JSAPI",
  171. // "transaction_id":"4200000479202004070804485027"
  172. // }
  173. // { // 支付宝支付成功回调参数
  174. // "gmt_create":"2020-04-10 19:15:00",
  175. // "charset":"utf-8",
  176. // "seller_email":"xptech@qq.com",
  177. // "subject":"\u5546\u57ce\u8ba2\u5355\u652f\u4ed8",
  178. // "sign":"AOiYZ1a2mEMOuIbHFCi6V6A0LJ97UMiHsCWgNdSU9dlzKFl15Ts8b0mL\/tN+Hhskl+94S3OUiNTBD3dD0Kv923SqaTWxNdj533PCdo2GDKsZIZgKbavnOvaccSKUdmQRE9KtmePPq9V9lFzEQvdUkKq1M8KAWO5K9LTy2iT2y5CUynpiu\/04GVzsTL9PqY+LDwqj6K+w7MgceWm1BWaFWg27AXIRw7wvsFckr3k9GGajgE2fufhoCYGYtGFbhGOp6ExtqS5RXBuPODOyRhBLpD8mwpOX38Oy0X+R4YQIrOi02dhqwPpvw79YjnvgXY3qZEQ66EdUsrT7EBdcPHK0Gw==",
  179. // "buyer_id":"2088902485164146",
  180. // "invoice_amount":"0.01",
  181. // "notify_id":"2020041000222191501064141414240102",
  182. // "fund_bill_list":"[{\"amount\":\"0.01\",\"fundChannel\":\"PCREDIT\"}]",
  183. // "notify_type":"trade_status_sync",
  184. // "trade_status":"TRADE_SUCCESS",
  185. // "receipt_amount":"0.01",
  186. // "buyer_pay_amount":"0.01",
  187. // "app_id":"2021001114666742",
  188. // "sign_type":"RSA2",
  189. // "seller_id":"2088721922277739",
  190. // "gmt_payment":"2020-04-10 19:15:00",
  191. // "notify_time":"2020-04-10 19:15:01",
  192. // "version":"1.0",
  193. // "out_trade_no":"202007144778322770017000",
  194. // "total_amount":"0.01",
  195. // "trade_no":"2020041022001464141443020240",
  196. // "auth_app_id":"2021001114666742",
  197. // "buyer_logon_id":"157***@163.com",
  198. // "point_amount":"0.00"
  199. // }
  200. // { // 支付宝退款成功(交易关闭)回调参数
  201. // "gmt_create": "2020-08-15 14:48:32",
  202. // "charset": "utf-8",
  203. // "seller_email": "xptech@qq.com",
  204. // "gmt_payment": "2020-08-15 14:48:32",
  205. // "notify_time": "2020-08-15 16:11:45",
  206. // "subject": "商城订单支付",
  207. // "gmt_refund": "2020-08-15 16:11:45.140",
  208. // "sign": "b6ArkgzLIRteRL9FMGC6i\/jf6VwFYQbaGDGRx002W+pdmN5q9+O4edZ3ALF74fYaijSl9ksNr0dKdvanu3uYxBTcd\/GIS4N1CWzmCOv6pzgx5rO\/YvGoHLM3Yop0GKKuMxmnNsZ6jhYKEY7SYD3Y0L6PU9ZMdHV7yIiVj+zZmbKzUgK9MPDCEXs+nzpNAiSM8GTqYRSUvKobAK68hswG2k1QIcqr5z+ZmVYa\/nHHkoC9qXt5zwyGi4P+2eOsr6V2PjA3x8qqe7TN5aI1DeoZD5KqHPYYaYF17J2q6YPlgl3WUl1RhE7H86bivB1fIuYEv\/3+JR74WN\/o7krGw1RPHg==",
  209. // "out_biz_no": "R202004114414846255015300",
  210. // "buyer_id": "2088902485164146",
  211. // "version": "1.0",
  212. // "notify_id": "2020081500222161145064141453349793",
  213. // "notify_type": "trade_status_sync",
  214. // "out_trade_no": "202002460317545607015300",
  215. // "total_amount": "0.01",
  216. // "trade_status": "TRADE_CLOSED",
  217. // "refund_fee": "0.01",
  218. // "trade_no": "2020081522001464141438570535",
  219. // "auth_app_id": "2021001114666742",
  220. // "buyer_logon_id": "157***@163.com",
  221. // "gmt_close": "2020-08-15 16:11:45",
  222. // "app_id": "2021001114666742",
  223. // "sign_type": "RSA2",
  224. // "seller_id": "2088721922277739"
  225. // }
  226. try {
  227. $out_trade_no = $data['out_trade_no'];
  228. $out_refund_no = $data['out_biz_no'] ?? '';
  229. // 判断是否是支付宝退款(支付宝退款成功会通知该接口)
  230. if ($payment == 'alipay' // 支付宝支付
  231. && $data['notify_type'] == 'trade_status_sync' // 同步交易状态
  232. && $data['trade_status'] == 'TRADE_CLOSED' // 交易关闭
  233. && $out_refund_no // 退款单号
  234. ) {
  235. // 退款回调
  236. $this->refundFinish($out_trade_no, $out_refund_no);
  237. return $pay->success()->send();
  238. }
  239. // 判断支付宝微信是否是支付成功状态,如果不是,直接返回响应
  240. if ($payment == 'alipay' && $data['trade_status'] != 'TRADE_SUCCESS') {
  241. // 不是交易成功的通知,直接返回成功
  242. return $pay->success()->send();
  243. }
  244. if ($payment == 'wechat' && ($data['result_code'] != 'SUCCESS' || $data['return_code'] != 'SUCCESS')) {
  245. // 微信交易未成功,返回 false,让微信再次通知
  246. return false;
  247. }
  248. // 支付成功流程
  249. $pay_fee = $payment == 'alipay' ? $data['total_amount'] : $data['total_fee'] / 100;
  250. //你可以在此编写订单逻辑
  251. $order = Order::where('order_sn', $out_trade_no)->find();
  252. if (!$order || $order->status > 0) {
  253. // 订单不存在,或者订单已支付
  254. return $pay->success()->send();
  255. }
  256. Db::transaction(function () use ($order, $data, $payment, $platform, $pay_fee) {
  257. $notify = [
  258. 'order_sn' => $data['out_trade_no'],
  259. 'transaction_id' => $payment == 'alipay' ? $data['trade_no'] : $data['transaction_id'],
  260. 'notify_time' => date('Y-m-d H:i:s', strtotime($data['time_end'])),
  261. 'buyer_email' => $payment == 'alipay' ? $data['buyer_logon_id'] : $data['openid'],
  262. 'payment_json' => json_encode($data->all()),
  263. 'pay_fee' => $pay_fee,
  264. 'pay_type' => $payment // 支付方式
  265. ];
  266. $order->paymentProcess($order, $notify);
  267. });
  268. return $pay->success()->send();
  269. } catch (\Exception $e) {
  270. Log::write('notifyx-error:' . json_encode($e->getMessage()));
  271. }
  272. });
  273. return $result;
  274. }
  275. /**
  276. * 退款成功回调
  277. */
  278. public function notifyr()
  279. {
  280. Log::write('notifyreturn-comein:');
  281. $payment = $this->request->param('payment');
  282. $platform = $this->request->param('platform');
  283. $pay = new \addons\shopro\library\PayService($payment, $platform);
  284. $result = $pay->notifyRefund(function ($data, $pay) use ($payment, $platform) {
  285. Log::write('notifyr-result:' . json_encode($data));
  286. try {
  287. $out_refund_no = $data['out_refund_no'];
  288. $out_trade_no = $data['out_trade_no'];
  289. // 退款
  290. $this->refundFinish($out_trade_no, $out_refund_no);
  291. return $pay->success()->send();
  292. } catch (\Exception $e) {
  293. Log::write('notifyreturn-error:' . json_encode($e->getMessage()));
  294. }
  295. });
  296. return $result;
  297. }
  298. private function refundFinish($out_trade_no, $out_refund_no) {
  299. $order = Order::where('order_sn', $out_trade_no)->find();
  300. $refundLog = \app\admin\model\shopro\order\RefundLog::where('refund_sn', $out_refund_no)->find();
  301. if (!$order || !$refundLog || $refundLog->status != 0) {
  302. // 订单不存在,或者订单已退款
  303. return true;
  304. }
  305. $item = \app\admin\model\shopro\order\OrderItem::where('id', $refundLog->order_item_id)->find();
  306. Db::transaction(function () use ($order, $item, $refundLog) {
  307. \app\admin\model\shopro\order\Order::refundFinish($order, $item, $refundLog);
  308. });
  309. return true;
  310. }
  311. }