PaymentController.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. <?php
  2. namespace App\Http\Controllers;
  3. use App\Components\Helpers;
  4. use App\Components\Yzy;
  5. use App\Components\Trimepay;
  6. use App\Http\Models\Coupon;
  7. use App\Http\Models\Goods;
  8. use App\Http\Models\Order;
  9. use App\Http\Models\Payment;
  10. use App\Http\Models\PaymentCallback;
  11. use Illuminate\Http\Request;
  12. use Response;
  13. use Redirect;
  14. use Log;
  15. use DB;
  16. use Auth;
  17. /**
  18. * 支付控制器
  19. *
  20. * Class PaymentController
  21. *
  22. * @package App\Http\Controllers
  23. */
  24. class PaymentController extends Controller
  25. {
  26. protected static $systemConfig;
  27. function __construct()
  28. {
  29. self::$systemConfig = Helpers::systemConfig();
  30. }
  31. // 创建支付单
  32. public function create(Request $request)
  33. {
  34. $goods_id = intval($request->get('goods_id'));
  35. $coupon_sn = $request->get('coupon_sn');
  36. $goods = Goods::query()->where('is_del', 0)->where('status', 1)->where('id', $goods_id)->first();
  37. if (!$goods) {
  38. return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败:商品或服务已下架']);
  39. }
  40. // 判断是否开启有赞云支付
  41. if (!self::$systemConfig['is_youzan']&&!self::$systemConfig['is_trimepay']) {
  42. return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败:系统并未开启在线支付功能']);
  43. }
  44. // 判断是否存在同个商品的未支付订单
  45. $existsOrder = Order::query()->where('status', 0)->where('user_id', Auth::user()->id)->where('goods_id', $goods_id)->exists();
  46. if ($existsOrder) {
  47. return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败:尚有未支付的订单,请先去支付']);
  48. }
  49. // 限购控制
  50. $strategy = self::$systemConfig['goods_purchase_limit_strategy'];
  51. if ($strategy == 'all' || ($strategy == 'package' && $goods->type == 2) || ($strategy == 'free' && $goods->price == 0) || ($strategy == 'package&free' && ($goods->type == 2 || $goods->price == 0))) {
  52. $noneExpireOrderExist = Order::query()->where('status', '>=', 0)->where('is_expire', 0)->where('user_id', Auth::user()->id)->where('goods_id', $goods_id)->exists();
  53. if ($noneExpireOrderExist) {
  54. return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败:商品不可重复购买']);
  55. }
  56. }
  57. // 单个商品限购
  58. if ($goods->is_limit == 1) {
  59. $noneExpireOrderExist = Order::query()->where('status', '>=', 0)->where('user_id', Auth::user()->id)->where('goods_id', $goods_id)->exists();
  60. if ($noneExpireOrderExist) {
  61. return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败:此商品每人限购1次']);
  62. }
  63. }
  64. // 使用优惠券
  65. if ($coupon_sn) {
  66. $coupon = Coupon::query()->where('status', 0)->where('is_del', 0)->whereIn('type', [1, 2])->where('sn', $coupon_sn)->first();
  67. if (!$coupon) {
  68. return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败:优惠券不存在']);
  69. }
  70. // 计算实际应支付总价
  71. $amount = $coupon->type == 2 ? $goods->price * $coupon->discount / 10 : $goods->price - $coupon->amount;
  72. $amount = $amount > 0 ? $amount : 0;
  73. } else {
  74. $amount = $goods->price;
  75. }
  76. // 价格异常判断
  77. if ($amount < 0) {
  78. return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败:订单总价异常']);
  79. } elseif ($amount == 0) {
  80. return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败:订单总价为0,无需使用在线支付']);
  81. }
  82. // 验证账号是否存在有效期更长的套餐
  83. if ($goods->type == 2) {
  84. $existOrderList = Order::query()
  85. ->with(['goods'])
  86. ->whereHas('goods', function ($q) {
  87. $q->where('type', 2);
  88. })
  89. ->where('user_id', Auth::user()->id)
  90. ->where('is_expire', 0)
  91. ->where('status', 2)
  92. ->get();
  93. foreach ($existOrderList as $vo) {
  94. if ($vo->goods->days > $goods->days) {
  95. return Response::json(['status' => 'fail', 'data' => '', 'message' => '支付失败:您已存在有效期更长的套餐,只能购买流量包']);
  96. }
  97. }
  98. }
  99. DB::beginTransaction();
  100. try {
  101. $orderSn = date('ymdHis') . mt_rand(100000, 999999);
  102. $sn = makeRandStr(12);
  103. // 生成订单
  104. $order = new Order();
  105. $order->order_sn = $orderSn;
  106. $order->user_id = Auth::user()->id;
  107. $order->goods_id = $goods_id;
  108. $order->coupon_id = !empty($coupon) ? $coupon->id : 0;
  109. $order->origin_amount = $goods->price;
  110. $order->amount = $amount;
  111. $order->expire_at = date("Y-m-d H:i:s", strtotime("+" . $goods->days . " days"));
  112. $order->is_expire = 0;
  113. if(self::$systemConfig['is_youzan']){
  114. $order->pay_way = 2;
  115. }else if(self::$systemConfig['is_trimepay']){
  116. $order->pay_way = 3;
  117. }
  118. $order->status = 0;
  119. $order->save();
  120. // 生成支付单
  121. if(self::$systemConfig['is_youzan']){
  122. $yzy = new Yzy();
  123. $result = $yzy->createQrCode($goods->name, $amount * 100, $orderSn);
  124. if (isset($result['error_response'])) {
  125. Log::error('【有赞云】创建二维码失败:' . $result['error_response']['msg']);
  126. throw new \Exception($result['error_response']['msg']);
  127. }
  128. }else if(self::$systemConfig['is_trimepay']){
  129. $trimepay = new Trimepay(self::$systemConfig['trimepay_appid'], self::$systemConfig['trimepay_appsecret']);
  130. $result = $trimepay->pay('WEPAY_QR', $orderSn, $amount, self::$systemConfig['website_url'].'/api/trimepay', self::$systemConfig['website_url']);
  131. if ($result['code']!==0) {
  132. Log::error('【Trimepay】创建二维码失败:' . $result['msg']);
  133. throw new \Exception($result['msg']);
  134. }
  135. }
  136. $payment = new Payment();
  137. $payment->sn = $sn;
  138. $payment->user_id = Auth::user()->id;
  139. $payment->oid = $order->oid;
  140. $payment->order_sn = $orderSn;
  141. $payment->pay_way = 1;
  142. $payment->amount = $amount;
  143. if(self::$systemConfig['is_youzan']){
  144. $payment->qr_id = $result['response']['qr_id'];
  145. $payment->qr_url = $result['response']['qr_url'];
  146. $payment->qr_code = $result['response']['qr_code'];
  147. $payment->qr_local_url = $this->base64ImageSaver($result['response']['qr_code']);
  148. }else if(self::$systemConfig['is_trimepay']){
  149. $payment->qr_url = $result['data'];
  150. $payment->qr_code = 'https://www.zhihu.com/qrcode?url='.$result['data'];
  151. $payment->qr_local_url = 'https://www.zhihu.com/qrcode?url='.$result['data'];
  152. }
  153. $payment->status = 0;
  154. $payment->save();
  155. // 优惠券置为已使用
  156. if (!empty($coupon)) {
  157. if ($coupon->usage == 1) {
  158. $coupon->status = 1;
  159. $coupon->save();
  160. }
  161. Helpers::addCouponLog($coupon->id, $goods_id, $order->oid, '在线支付使用');
  162. }
  163. DB::commit();
  164. return Response::json(['status' => 'success', 'data' => $sn, 'message' => '创建订单成功,正在转到付款页面,请稍后']);
  165. } catch (\Exception $e) {
  166. DB::rollBack();
  167. Log::error('创建支付订单失败:' . $e->getMessage());
  168. return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建订单失败:' . $e->getMessage()]);
  169. }
  170. }
  171. // 支付单详情
  172. public function detail(Request $request, $sn)
  173. {
  174. if (empty($sn)) {
  175. return Redirect::to('services');
  176. }
  177. $payment = Payment::query()->with(['order', 'order.goods'])->where('sn', $sn)->where('user_id', Auth::user()->id)->first();
  178. if (!$payment) {
  179. return Redirect::to('services');
  180. }
  181. $order = Order::query()->where('oid', $payment->oid)->first();
  182. if (!$order) {
  183. \Session::flash('errorMsg', '订单不存在');
  184. return Response::view('payment/' . $sn);
  185. }
  186. $view['payment'] = $payment;
  187. $view['website_logo'] = self::$systemConfig['website_logo'];
  188. $view['website_analytics'] = self::$systemConfig['website_analytics'];
  189. $view['website_customer_service'] = self::$systemConfig['website_customer_service'];
  190. return Response::view('payment.detail', $view);
  191. }
  192. // 获取订单支付状态
  193. public function getStatus(Request $request)
  194. {
  195. $sn = $request->get('sn');
  196. if (empty($sn)) {
  197. return Response::json(['status' => 'fail', 'data' => '', 'message' => '请求失败']);
  198. }
  199. $payment = Payment::query()->where('sn', $sn)->where('user_id', Auth::user()->id)->first();
  200. if (!$payment) {
  201. return Response::json(['status' => 'error', 'data' => '', 'message' => '支付失败']);
  202. } elseif ($payment->status > 0) {
  203. return Response::json(['status' => 'success', 'data' => '', 'message' => '支付成功']);
  204. } elseif ($payment->status < 0) {
  205. return Response::json(['status' => 'error', 'data' => '', 'message' => '订单超时未支付,已自动关闭']);
  206. } else {
  207. return Response::json(['status' => 'fail', 'data' => '', 'message' => '等待支付']);
  208. }
  209. }
  210. // 有赞云回调日志
  211. public function callbackList(Request $request)
  212. {
  213. $status = $request->get('status', 0);
  214. $query = PaymentCallback::query();
  215. if ($status) {
  216. $query->where('status', $status);
  217. }
  218. $view['list'] = $query->orderBy('id', 'desc')->paginate(10);
  219. return Response::view('payment.callbackList', $view);
  220. }
  221. }