Files
ProxyPanel/app/Http/Controllers/PaymentController.php
兔姬桑 0e2e5f61dd 2.3 版本 功能添加与优化
加入新支付方式 PayJS ← 未实装,后续版本实装。目前本人还卡在他们的账号注册上面,没法测试代码o(╥﹏╥)o
用户注册等地方添加新字段: 昵称; 用于后续邮件,页面显示;还需要更多优化与应用;
用户头像获取与昵称获取; 前提是用户有填写QQ信息或者注册邮箱为正常的QQ邮箱;
添加 维护模式; 模式开启后,用户界面将自动导到维护界面,管理后台正常运行。哀悼日网站可以添加大厂一样的哀悼维护了 3.1 维护模式定时关闭功能; 3.2 维护模式自定义提示语;
Ping检测与定时记录; 4.1 添加Ping记录界面; 4.2 节点界面添加检测独立Ping按钮; 4.3 针对用户界面添加Ping参考信息;
余额充值自定义功能
部分界面js显示优化
模块化页面;
2020-08-05 03:20:25 +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 = $balance? 0 : $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);
}
}