Files
ProxyPanel/app/Http/Controllers/PaymentController.php
Bretton d1581f0e33 1.3
功能添加/重置
1. 商品限购功能重置 - 商品为0时下架
2. 添加用户付费自行重置套餐流量 - 可规定价格让用户重置或不开启功能
3. 邮箱注册增加白名单 - 白名单为用户只能使用固定邮箱后缀注册
4. 节点TCP和ICMP检测 - 节点阻断检测
5. 优惠券规则添加 - 满XX元抵XX,或者满XX元打折

页面添加与修改
1. 节点信息简化
2. 系统页面排版优化
3. 注册页面全新的TOS和AUP显示
4. 添加订阅记录查询页面
5. 人性化删除确认页

BUG修复和优化
1. 流量重置日相关代码统一
2. 众多代码简化
3. auth相关css导致的宽度变动bug修复
2020-08-05 03:20:19 +08:00

250 lines
9.7 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\Helpers;
use App\Components\Yzy;
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 Auth;
use DB;
use Exception;
use Illuminate\Http\Request;
use Log;
use Payment\Client\Charge;
use Response;
use Validator;
/**
* 支付控制器
*
* Class PaymentController
*
* @package App\Http\Controllers
*/
class PaymentController extends Controller
{
protected static $systemConfig;
function __construct()
{
self::$systemConfig = Helpers::systemConfig();
}
// 创建支付单
public function create(Request $request)
{
$goods_id = $request->input('goods_id');
$coupon_sn = $request->input('coupon_sn');
$goods = Goods::query()->where('status', 1)->where('id', $goods_id)->first();
if(!$goods){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败:商品或服务已下架']);
}
// 判断是否开启有赞云支付
if(!self::$systemConfig['is_youzan'] && !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){
return Response::json(['status' => 'fail', 'data' => '', 'message' => '创建支付单失败订单总价为0无需使用在线支付']);
}
// 验证账号是否存在有效期更长的套餐
if($goods->type == 2){
$existOrderList = Order::uid()->with(['goods'])->whereHas('goods', function($q){
$q->where('type', 2);
})->where('is_expire', 0)->where('status', 2)->get();
foreach($existOrderList as $vo){
if($vo->goods->days > $goods->days){
return Response::json(['status' => 'info', 'title' => '套餐冲突', 'message' => '是否将本次套餐存为 【预支付】?套餐会在已有套餐失效后生效,或者您可以手动激活套餐']);
}
}
}
DB::beginTransaction();
try{
$orderSn = date('ymdHis').mt_rand(100000, 999999);
$sn = makeRandStr(12);
// 支付方式
if(self::$systemConfig['is_youzan']){
$pay_way = 2;
}elseif(self::$systemConfig['is_alipay']){
$pay_way = 4;
}elseif(self::$systemConfig['is_f2fpay']){
$pay_way = 5;
}
// 生成订单
$order = new Order();
$order->order_sn = $orderSn;
$order->user_id = Auth::user()->id;
$order->goods_id = $goods_id;
$order->coupon_id = !empty($coupon)? $coupon->id : 0;
$order->origin_amount = $goods->price;
$order->amount = $amount;
$order->expire_at = date("Y-m-d H:i:s", strtotime("+".$goods->days." days"));
$order->is_expire = 0;
$order->pay_way = $pay_way;
$order->status = 0;
$order->save();
// 生成支付单
if(self::$systemConfig['is_youzan']){
$yzy = new Yzy();
$result = $yzy->createQrCode($goods->name, $amount*100, $orderSn);
if(isset($result['error_response'])){
Log::error('【有赞云】创建二维码失败:'.$result['error_response']['msg']);
throw new Exception($result['error_response']['msg']);
}
}elseif(self::$systemConfig['is_alipay']){
$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']){
// TODOgoods表里增加一个字段用于自定义商品付款时展示的商品名称
// TODO这里增加一个随机商品列表根据goods的价格随机取值
$result = Charge::run("ali_qr", ['use_sandbox' => FALSE, "partner" => self::$systemConfig['f2fpay_app_id'], 'app_id' => self::$systemConfig['f2fpay_app_id'], 'sign_type' => 'RSA2', 'ali_public_key' => self::$systemConfig['f2fpay_public_key'], 'rsa_private_key' => self::$systemConfig['f2fpay_private_key'], 'notify_url' => self::$systemConfig['website_url']."/api/f2fpay", // 异步回调接口
'return_url' => self::$systemConfig['website_url'], 'return_raw' => FALSE], ['body' => '', 'subject' => self::$systemConfig['f2fpay_subject_name'], 'order_no' => $orderSn, 'amount' => $amount,]);
}
$payment = new Payment();
$payment->sn = $sn;
$payment->user_id = Auth::user()->id;
$payment->oid = $order->oid;
$payment->order_sn = $orderSn;
$payment->pay_way = 1;
$payment->amount = $amount;
if(self::$systemConfig['is_youzan']){
$payment->qr_id = $result['response']['qr_id'];
$payment->qr_url = $result['response']['qr_url'];
$payment->qr_code = $result['response']['qr_code'];
$payment->qr_local_url = $this->base64ImageSaver($result['response']['qr_code']);
}elseif(self::$systemConfig['is_alipay']){
$payment->qr_code = $result;
}elseif(self::$systemConfig['is_f2fpay']){
$payment->qr_code = $result;
$payment->qr_url = 'http://qr.topscan.com/api.php?text='.$result.'&bg=ffffff&fg=000000&pt=1c73bd&m=10&w=400&el=1&inpt=1eabfc&logo=https://t.alipayobjects.com/tfscom/T1Z5XfXdxmXXXXXXXX.png';
$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(self::$systemConfig['is_alipay']){ // Alipay返回支付信息
return Response::json(['status' => 'success', 'data' => $result, 'message' => '创建订单成功,正在转到付款页面,请稍后']);
}else{
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()]);
}
}
// 支付单详情
public function detail(Request $request, $sn)
{
$view['payment'] = Payment::uid()->with(['order', 'order.goods'])->where('sn', $sn)->firstOrFail();
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->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);
}
}