Files
ProxyPanel/app/Http/Controllers/PaymentController.php
兔姬桑 346e841ea0 修复余额方面的问题
修复使用余额购买套餐,扣除余额后,没有实际处理订单的问题;
修复余额订单中支付方式记录类型错误的问题;
修复商品订单页面orderList无法显示的问题;
2020-08-05 03:20:31 +08:00

335 lines
13 KiB
PHP
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<?php
namespace App\Http\Controllers;
use App\Components\AlipaySubmit;
use App\Components\Callback;
use App\Components\Helpers;
use App\Http\Models\Coupon;
use App\Http\Models\Goods;
use App\Http\Models\Order;
use App\Http\Models\Payment;
use App\Http\Models\PaymentCallback;
use App\Http\Models\User;
use Auth;
use DB;
use Exception;
use Illuminate\Http\Request;
use InvalidArgumentException;
use Log;
use Payment\Client;
use Payment\Exceptions\ClassNotFoundException;
use Payment\Exceptions\GatewayException;
use Response;
use Validator;
use Xhat\Payjs\Payjs;
/**
* 支付控制器
*
* Class PaymentController
*
* @package App\Http\Controllers
*/
class PaymentController extends Controller
{
use Callback;
// 创建支付订单
public function create(Request $request)
{
$goods_id = $request->input('goods_id');
$coupon_sn = $request->input('coupon_sn');
$pay_type = $request->input('pay_type');
$balance = $request->input('amount');
$goods = Goods::query()->where('status', 1)->where('id', $goods_id)->first();
if(isset($balance)){
if(!is_numeric($balance) || $balance <= 0){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '充值余额不合规']);
}
$amount = $balance;
}elseif(isset($goods_id) && isset($pay_type)){
if(!$goods){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '订单创建失败:商品或服务已下架']);
}
// 是否有生效的套餐
$activePlan = Order::uid()->with(['goods'])->whereHas('goods', function($q){ $q->where('type', 2); })->where('status', 2)->where('is_expire', 0)->doesntExist();
//无生效套餐,禁止购买加油包
if($goods->type == 1 && $activePlan){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '购买加油包前,请先购买套餐']);
}
//非余额付款下,检查对应的在线支付是否开启
if($pay_type != 1){
// 判断是否开启在线支付
if(!self::$systemConfig['is_alipay'] && !self::$systemConfig['is_f2fpay']){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '订单创建失败:系统并未开启在线支付功能']);
}
// 判断是否存在同个商品的未支付订单
$existsOrder = Order::uid()->where('status', 0)->where('goods_id', $goods_id)->exists();
if($existsOrder){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '订单创建失败:尚有未支付的订单,请先去支付']);
}
}
// 单个商品限购
if($goods->limit_num){
$count = Order::uid()->where('status', '>=', 0)->where('goods_id', $goods_id)->count();
if($count >= $goods->limit_num){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '此商品/服务限购'.$goods->limit_num.'次,您已购买'.$count.'次']);
}
}
// 使用优惠券
if($coupon_sn){
$coupon = Coupon::query()->where('status', 0)->whereIn('type', [1, 2])->where('sn', $coupon_sn)->first();
if(!$coupon){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '订单创建失败:优惠券不存在']);
}
// 计算实际应支付总价
$amount = $coupon->type == 2? $goods->price*$coupon->discount/10 : $goods->price-$coupon->amount;
$amount = $amount > 0? round($amount, 2) : 0; // 四舍五入保留2位小数避免无法正常创建订单
}else{
$amount = $goods->price;
}
// 价格异常判断
if($amount < 0){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '订单创建失败:订单总价异常']);
}elseif($amount == 0 && $pay_type != 1){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '订单创建失败订单总价为0无需使用在线支付']);
}
// 验证账号余额是否充足
if($pay_type == 1 && Auth::user()->balance < $amount){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '您的余额不足,请先充值']);
}
}
try{
DB::beginTransaction();
$orderSn = date('ymdHis').mt_rand(100000, 999999);
// 生成订单
$order = new Order();
$order->order_sn = $orderSn;
$order->user_id = Auth::user()->id;
$order->goods_id = $balance? -1 : $goods_id;
$order->coupon_id = !empty($coupon)? $coupon->id : 0;
$order->origin_amount = $balance? : $goods->price;
$order->amount = $amount;
$order->expire_at = $balance? NULL : date("Y-m-d H:i:s", strtotime("+".$goods->days." days"));
$order->is_expire = 0;
$order->pay_way = $pay_type;
$order->status = 0;
$order->save();
// 生成支付单
if($pay_type == 1){
// 扣余额
User::query()->where('id', Auth::user()->id)->decrement('balance', $amount*100);
// 记录余额操作日志
$this->addUserBalanceLog(Auth::user()->id, $order->oid, Auth::user()->balance, Auth::user()->balance-$amount, -1*$amount, '购买商品:'.$goods->name);
$data = [];
$data['out_trade_no'] = $orderSn;
$this->tradePaid($data, 1);
}else{
if(self::$systemConfig['is_alipay'] && $pay_type == 4){
$pay_way = 2;
$parameter = [
"service" => "create_forex_trade", // WAP:create_forex_trade_wap ,即时到帐:create_forex_trade
"partner" => self::$systemConfig['alipay_partner'],
"notify_url" => self::$systemConfig['website_url']."/api/alipay", // 异步回调接口
"return_url" => self::$systemConfig['website_url'],
"out_trade_no" => $orderSn, // 订单号
"subject" => "Package", // 订单名称
//"total_fee" => $amount, // 金额
"rmb_fee" => $amount, // 使用RMB标价不再使用总金额
"body" => "", // 商品描述,可为空
"currency" => self::$systemConfig['alipay_currency'], // 结算币种
"product_code" => "NEW_OVERSEAS_SELLER",
"_input_charset" => "utf-8"
];
// 建立请求
$alipaySubmit = new AlipaySubmit(self::$systemConfig['alipay_sign_type'], self::$systemConfig['alipay_partner'], self::$systemConfig['alipay_key'], self::$systemConfig['alipay_private_key']);
$result = $alipaySubmit->buildRequestForm($parameter, "post", "确认");
}elseif(self::$systemConfig['is_f2fpay'] && $pay_type == 5){
$pay_way = 2;
// TODOgoods表里增加一个字段用于自定义商品付款时展示的商品名称
// TODO这里增加一个随机商品列表根据goods的价格随机取值
$aliConfig = [
'use_sandbox' => FALSE, // 是否使用沙盒模式
'app_id' => self::$systemConfig['f2fpay_app_id'],
'sign_type' => 'RSA2', // RSA RSA2
'ali_public_key' => self::$systemConfig['f2fpay_public_key'],
'rsa_private_key' => self::$systemConfig['f2fpay_private_key'],
'limit_pay' => [
//'balance',// 余额
//'moneyFund',// 余额宝
//'debitCardExpress',// 借记卡快捷
//'creditCard',//信用卡
//'creditCardExpress',// 信用卡快捷
//'creditCardCartoon',//信用卡卡通
//'credit_group',// 信用支付类型(包含信用卡卡通、信用卡快捷、花呗、花呗分期)
], // 用户不可用指定渠道支付当有多个渠道时用“,”分隔
'notify_url' => self::$systemConfig['website_url']."/api/f2fpay",
'return_url' => self::$systemConfig['website_url'],
'fee_type' => 'CNY', // 货币类型 当前仅支持该字段
];
$payData = [
'body' => '',
'subject' => self::$systemConfig['f2fpay_subject_name']? : self::$systemConfig['website_name'],
'trade_no' => $orderSn,
'time_expire' => time()+9000, // 表示必须 1000s 内付款
'amount' => $amount, // 单位为元 ,最小为0.01
];
try{
$client = new Client(Client::ALIPAY, $aliConfig);
$result = $client->pay(Client::ALI_CHANNEL_QR, $payData);
} catch(InvalidArgumentException $e){
Log::error("【支付宝当面付】输入信息错误: ".$e->getMessage());
exit;
} catch(GatewayException $e){
Log::error("【支付宝当面付】建立支付错误: ".$e->getMessage()." | ".var_dump($e->getRaw()));
exit;
} catch(ClassNotFoundException $e){
Log::error("【支付宝当面付】未知类型: ".$e->getMessage());
exit;
} catch(Exception $e){
Log::error("【支付宝当面付】错误: ".$e->getMessage());
exit;
}
}elseif($pay_type == 6){
$pay_way = 1;
// 配置通信参数
$config = [
'mchid' => self::$systemConfig['payjs_mch_id'],
'key' => self::$systemConfig['payjs_key'],
];
// 初始化
$payjs = new Payjs($config);
$data = [
'body' => '',
'total_fee' => $amount,
'out_trade_no' => $orderSn,
'notify_url' => self::$systemConfig['website_url']."/api/payjs",
];
$result = $payjs->native($data);
if(!$result['return_code']){
Log::error("【PayJs】错误: ".$result['return_msg']);
return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败:支付渠道暂时无法提供支付信息!']);
}
}else{
return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败:未知支付类型']);
}
$sn = makeRandStr(12);
$payment = new Payment();
$payment->sn = $sn;
$payment->user_id = Auth::user()->id;
$payment->oid = $order->oid;
$payment->order_sn = $orderSn;
$payment->pay_way = $pay_way? : 1;
$payment->amount = $amount;
if($pay_type == 4){
$payment->qr_code = $result;
}elseif($pay_type == 5){
$payment->qr_code = $result['qr_code'];
$payment->qr_url = 'http://qr.topscan.com/api.php?text='.$result['qr_code'].'&bg=ffffff&fg=000000&pt=1c73bd&m=10&w=400&el=1&inpt=1eabfc&logo=https://t.alipayobjects.com/tfscom/T1Z5XfXdxmXXXXXXXX.png'; //后备https://cli.im/api/qrcode/code?text=".$result['qr_code']."&mhid=5EfGCwztyckhMHcmI9ZcOKs
$payment->qr_local_url = $payment->qr_url;
}elseif($pay_type == 6){
$payment->qr_code = $result['code_url'];
$payment->qr_url = $result['qrcode'];
$payment->qr_local_url = $payment->qr_url;
}
$payment->status = 0;
$payment->save();
}
// 优惠券置为已使用
if(!empty($coupon)){
if($coupon->usage == 1){
$coupon->status = 1;
$coupon->save();
}
Helpers::addCouponLog($coupon->id, $goods_id, $order->oid, '订单支付使用');
}
DB::commit();
if($pay_type == 1){
return Response::json(['status' => 'success', 'data' => '', 'message' => '支付成功']);
}elseif($pay_type == 4){ // Alipay返回支付信息
return Response::json(['status' => 'success', 'data' => $result, 'message' => '创建订单成功,正在转到付款页面,请稍后']);
}elseif($pay_type == 5){
return Response::json(['status' => 'success', 'data' => $sn, 'message' => '创建订单成功,正在转到付款页面,请稍后']);
}
} catch(Exception $e){
DB::rollBack();
Log::error('创建支付订单失败:'.$e->getMessage());
return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建订单失败:'.$e->getMessage()]);
}
return Response::json(['status' => 'fail', 'data' => '', 'message' => '未知错误']);
}
// 支付单详情
public function detail($sn)
{
$payment = Payment::uid()->with(['order', 'order.goods'])->where('sn', $sn)->firstOrFail();
$view['payment'] = $payment;
$view['name'] = $payment->order->goods? $payment->order->goods->name : '余额充值';
$view['days'] = $payment->order->goods? $payment->order->goods->days : 0;
return Response::view('payment.detail', $view);
}
// 获取订单支付状态
public function getStatus(Request $request)
{
$validator = Validator::make($request->all(), ['sn' => 'required|exists:payment,sn'], ['sn.required' => '请求失败缺少sn', 'sn.exists' => '支付失败:支付单不存在']);
if($validator->fails()){
return Response::json(['status' => 'error', 'data' => '', 'message' => $validator->getMessageBag()->first()]);
}
$payment = Payment::uid()->where('sn', $request->input('sn'))->first();
if($payment->status > 0){
return Response::json(['status' => 'success', 'data' => '', 'message' => '支付成功']);
}elseif($payment->status < 0){
return Response::json(['status' => 'error', 'data' => '', 'message' => '订单超时未支付,已自动关闭']);
}else{
return Response::json(['status' => 'fail', 'data' => '', 'message' => '等待支付']);
}
}
// 回调日志
public function callbackList(Request $request)
{
$status = $request->input('status', 0);
$query = PaymentCallback::query();
if(isset($status)){
$query->where('status', $status);
}
$view['list'] = $query->orderBy('id', 'desc')->paginate(10)->appends($request->except('page'));
return Response::view('payment.callbackList', $view);
}
}