PaymentController.php 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. <?php
  2. namespace App\Http\Controllers;
  3. use App\Components\Helpers;
  4. use App\Http\Controllers\Gateway\BitpayX;
  5. use App\Http\Controllers\Gateway\CodePay;
  6. use App\Http\Controllers\Gateway\EPay;
  7. use App\Http\Controllers\Gateway\F2Fpay;
  8. use App\Http\Controllers\Gateway\Local;
  9. use App\Http\Controllers\Gateway\PayJs;
  10. use App\Http\Controllers\Gateway\PayPal;
  11. use App\Models\Coupon;
  12. use App\Models\Goods;
  13. use App\Models\Order;
  14. use App\Models\Payment;
  15. use App\Models\PaymentCallback;
  16. use Auth;
  17. use Illuminate\Http\JsonResponse;
  18. use Illuminate\Http\Request;
  19. use Log;
  20. use Response;
  21. /**
  22. * 支付控制器
  23. *
  24. * Class PaymentController
  25. *
  26. * @package App\Http\Controllers
  27. */
  28. class PaymentController extends Controller
  29. {
  30. private static string $method;
  31. public static function notify(Request $request): int
  32. {
  33. self::$method = $request->input('method');
  34. Log::info(self::$method."回调接口[POST]:".self::$method.var_export($request->all(), true));
  35. self::getClient()->notify($request);
  36. return 0;
  37. }
  38. public static function getClient()
  39. {
  40. switch (self::$method) {
  41. case 'credit':
  42. return new Local();
  43. case 'f2fpay':
  44. return new F2Fpay();
  45. case 'codepay':
  46. return new Codepay();
  47. case 'payjs':
  48. return new PayJs();
  49. case 'bitpayx':
  50. return new BitpayX();
  51. case 'paypal':
  52. return new PayPal();
  53. case 'epay':
  54. return new EPay();
  55. default:
  56. Log::error("未知支付:".self::$method);
  57. return false;
  58. }
  59. }
  60. public static function getStatus(Request $request): JsonResponse
  61. {
  62. $payment = Payment::whereTradeNo($request->input('trade_no'))->first();
  63. if ($payment) {
  64. if ($payment->status === 1) {
  65. return Response::json(['status' => 'success', 'message' => '支付成功']);
  66. }
  67. if ($payment->status === -1) {
  68. return Response::json(['status' => 'error', 'message' => '订单超时未支付,已自动关闭']);
  69. }
  70. return Response::json(['status' => 'fail', 'message' => '等待支付']);
  71. }
  72. return Response::json(['status' => 'error', 'message' => '未知订单']);
  73. }
  74. // 创建支付订单
  75. public function purchase(Request $request)
  76. {
  77. $goods_id = $request->input('goods_id');
  78. $coupon_sn = $request->input('coupon_sn');
  79. self::$method = $request->input('method');
  80. $credit = $request->input('amount');
  81. $pay_type = $request->input('pay_type');
  82. $amount = 0;
  83. $goods = Goods::find($goods_id);
  84. // 充值余额
  85. if ($credit) {
  86. if (!is_numeric($credit) || $credit <= 0) {
  87. return Response::json(['status' => 'fail', 'message' => '充值余额不合规']);
  88. }
  89. $amount = $credit;
  90. // 购买服务
  91. } elseif ($goods_id && self::$method) {
  92. if (!$goods || !$goods->status) {
  93. return Response::json(['status' => 'fail', 'message' => '订单创建失败:商品已下架']);
  94. }
  95. $amount = $goods->price;
  96. // 是否有生效的套餐
  97. $activePlan = Order::userActivePlan()->doesntExist();
  98. // 无生效套餐,禁止购买加油包
  99. if ($goods->type === 1 && $activePlan) {
  100. return Response::json(['status' => 'fail', 'message' => '购买加油包前,请先购买套餐']);
  101. }
  102. //非余额付款下,检查在线支付是否开启
  103. if (self::$method !== 'credit') {
  104. // 判断是否开启在线支付
  105. if (!sysConfig('is_onlinePay')) {
  106. return Response::json(['status' => 'fail', 'message' => '订单创建失败:系统并未开启在线支付功能']);
  107. }
  108. // 判断是否存在同个商品的未支付订单
  109. if (Order::uid()->whereStatus(0)->exists()) {
  110. return Response::json(['status' => 'fail', 'message' => '订单创建失败:尚有未支付的订单,请先去支付']);
  111. }
  112. } elseif (Auth::getUser()->credit < $amount) { // 验证账号余额是否充足
  113. return Response::json(['status' => 'fail', 'message' => '您的余额不足,请先充值']);
  114. }
  115. // 单个商品限购
  116. if ($goods->limit_num) {
  117. $count = Order::uid()->where('status', '>=', 0)->whereGoodsId($goods_id)->count();
  118. if ($count >= $goods->limit_num) {
  119. return Response::json(['status' => 'fail', 'message' => '此商品限购'.$goods->limit_num.'次,您已购买'.$count.'次']);
  120. }
  121. }
  122. // 使用优惠券 TODO 代码整合至 CouponService
  123. if ($coupon_sn) {
  124. $coupon = Coupon::whereStatus(0)->whereIn('type', [1, 2])->whereSn($coupon_sn)->first();
  125. if (!$coupon) {
  126. return Response::json(['status' => 'fail', 'message' => '订单创建失败:优惠券不存在']);
  127. }
  128. // 计算实际应支付总价
  129. $amount = $coupon->type === 2 ? $goods->price * $coupon->value / 100 : $goods->price - $coupon->value;
  130. $amount = $amount > 0 ? round($amount, 2) : 0; // 四舍五入保留2位小数,避免无法正常创建订单
  131. }
  132. // 价格异常判断
  133. if ($amount < 0) {
  134. return Response::json(['status' => 'fail', 'message' => '订单创建失败:订单总价异常']);
  135. }
  136. if ($amount === 0 && self::$method !== 'credit') {
  137. return Response::json(['status' => 'fail', 'message' => '订单创建失败:订单总价为0,无需使用在线支付']);
  138. }
  139. }
  140. $orderSn = date('ymdHis').random_int(100000, 999999);
  141. // 生成订单
  142. $order = new Order();
  143. $order->order_sn = $orderSn;
  144. $order->user_id = Auth::id();
  145. $order->goods_id = $credit ? 0 : $goods_id;
  146. $order->coupon_id = $coupon->id ?? 0;
  147. $order->origin_amount = $credit ?: $goods->price;
  148. $order->amount = $amount;
  149. $order->is_expire = 0;
  150. $order->pay_type = $pay_type;
  151. $order->pay_way = self::$method;
  152. $order->status = 0;
  153. $order->save();
  154. // 使用优惠券,减少可使用次数
  155. if (!empty($coupon)) {
  156. if ($coupon->usable_times > 0) {
  157. Coupon::whereId($coupon->id)->decrement('usable_times', 1);
  158. }
  159. Helpers::addCouponLog('订单支付使用', $coupon->id, $goods_id, $order->id);
  160. }
  161. $request->merge(['id' => $order->id, 'type' => $pay_type, 'amount' => $amount]);
  162. // 生成支付单
  163. return self::getClient()->purchase($request);
  164. }
  165. public function close(Request $request): JsonResponse
  166. {
  167. $order = Order::find($request->input('id'));
  168. if ($order) {
  169. if (!$order->update(['status' => -1])) {
  170. return Response::json(['status' => 'fail', 'message' => '关闭订单失败']);
  171. }
  172. } else {
  173. return Response::json(['status' => 'fail', 'message' => '未找到订单']);
  174. }
  175. return Response::json(['status' => 'success', 'message' => '关闭订单成功']);
  176. }
  177. // 支付单详情
  178. public function detail($trade_no)
  179. {
  180. $payment = Payment::uid()->with(['order', 'order.goods'])->whereTradeNo($trade_no)->firstOrFail();
  181. $view['payment'] = $payment;
  182. $goods = $payment->order->goods;
  183. $view['name'] = $goods->name ?? '余额充值';
  184. $view['days'] = $goods->days ?? 0;
  185. $view['pay_type'] = $payment->order->pay_type_label ?: 0;
  186. $view['pay_type_icon'] = $payment->order->pay_type_icon;
  187. return view('user.payment', $view);
  188. }
  189. }