v2board/app/Http/Controllers/User/OrderController.php

527 lines
19 KiB
PHP
Raw Normal View History

2019-10-29 15:33:36 +08:00
<?php
2020-01-29 16:08:50 +08:00
namespace App\Http\Controllers\User;
2019-10-29 15:33:36 +08:00
use App\Http\Controllers\Controller;
2020-01-29 16:22:39 +08:00
use App\Http\Requests\User\OrderSave;
2020-03-17 20:16:55 +08:00
use App\Services\OrderService;
2020-03-17 20:00:46 +08:00
use App\Services\UserService;
2020-01-01 23:40:14 +08:00
use Illuminate\Http\Request;
2020-01-12 01:27:36 +08:00
use Illuminate\Support\Facades\Cache;
2019-12-27 01:59:15 +08:00
use Illuminate\Support\Facades\Log;
2020-01-01 23:40:14 +08:00
use Illuminate\Support\Facades\DB;
2019-10-29 15:33:36 +08:00
use App\Models\Order;
use App\Models\Plan;
use App\Models\User;
2020-01-01 17:55:16 +08:00
use App\Models\Coupon;
2019-10-29 15:33:36 +08:00
use App\Utils\Helper;
use Omnipay\Omnipay;
use Stripe\Stripe;
use Stripe\Source;
2019-12-27 01:50:26 +08:00
use Library\BitpayX;
2020-01-14 00:29:59 +08:00
use Library\PayTaro;
2019-10-29 15:33:36 +08:00
class OrderController extends Controller
{
2020-01-11 13:36:52 +08:00
public function fetch(Request $request)
{
2020-01-27 15:49:44 +08:00
$model = Order::where('user_id', $request->session()->get('id'))
->orderBy('created_at', 'DESC');
2020-01-29 16:08:50 +08:00
if ($request->input('status') !== null) {
2020-01-27 15:49:44 +08:00
$model->where('status', $request->input('status'));
}
$order = $model->get();
2019-10-29 15:33:36 +08:00
$plan = Plan::get();
2020-01-11 13:36:52 +08:00
for ($i = 0; $i < count($order); $i++) {
for ($x = 0; $x < count($plan); $x++) {
2019-10-29 15:33:36 +08:00
if ($order[$i]['plan_id'] === $plan[$x]['id']) {
$order[$i]['plan'] = $plan[$x];
}
}
}
return response([
'data' => $order
]);
}
2020-01-11 13:36:52 +08:00
public function details(Request $request)
{
2019-10-29 15:33:36 +08:00
$order = Order::where('user_id', $request->session()->get('id'))
->where('trade_no', $request->input('trade_no'))
->first();
if (!$order) {
abort(500, '订单不存在');
}
$order['plan'] = Plan::find($order->plan_id);
2020-01-12 02:08:06 +08:00
$order['try_out_plan_id'] = (int)config('v2board.try_out_plan_id');
2019-10-29 15:33:36 +08:00
if (!$order['plan']) {
abort(500, '订阅不存在');
}
return response([
'data' => $order
]);
}
2020-01-02 00:05:54 +08:00
2020-02-11 15:45:30 +08:00
private function isNotCompleteOrderByUserId($userId)
2020-01-11 13:36:52 +08:00
{
2020-02-11 15:45:30 +08:00
$order = Order::whereIn('status', [0, 1])
2020-01-02 00:11:42 +08:00
->where('user_id', $userId)
2020-01-02 00:05:54 +08:00
->first();
2020-01-02 00:20:31 +08:00
if (!$order) {
return false;
}
return true;
2020-01-02 00:05:54 +08:00
}
2020-01-11 13:36:52 +08:00
2020-02-17 01:00:12 +08:00
// surplus value
private function getSurplusValue(User $user)
2020-02-17 00:35:21 +08:00
{
2020-02-17 00:42:56 +08:00
$plan = Plan::find($user->plan_id);
2020-02-28 14:51:06 +08:00
switch ($plan->type) {
case 0: return $this->getSurplusValueByCycle($user, $plan);
case 1: return $this->getSurplusValueByOneTime($user, $plan);
}
}
private function getSurplusValueByOneTime(User $user, Plan $plan)
{
$trafficUnitPrice = 0;
$trafficUnitPrice = $plan->onetime_price / $plan->transfer_enable;
if ($user->discount && $trafficUnitPrice) {
$trafficUnitPrice = $trafficUnitPrice - ($trafficUnitPrice * $user->discount / 100);
}
$notUsedTrafficPrice = $plan->transfer_enable - (($user->u + $user->d) / 1073741824);
$result = $trafficUnitPrice * $notUsedTrafficPrice;
return $result > 0 ? $result : 0;
2020-02-28 14:51:06 +08:00
}
private function getSurplusValueByCycle(User $user, Plan $plan)
{
2020-02-17 12:25:19 +08:00
$dayPrice = 0;
2020-03-29 23:49:06 +08:00
$day = ($user->expired_at - time()) / 86400;
2020-02-17 00:35:21 +08:00
if ($plan->month_price) {
2020-03-29 23:49:06 +08:00
$dayPrice = $plan->month_price / $day;
2020-02-17 00:35:21 +08:00
} else if ($plan->quarter_price) {
2020-03-29 23:49:06 +08:00
$dayPrice = $plan->quarter_price / $day;
2020-02-17 00:35:21 +08:00
} else if ($plan->half_year_price) {
2020-03-29 23:49:06 +08:00
$dayPrice = $plan->half_year_price / $day;
2020-02-17 00:35:21 +08:00
} else if ($plan->year_price) {
2020-03-29 23:49:06 +08:00
$dayPrice = $plan->year_price / $day;
2020-02-17 00:35:21 +08:00
}
2020-02-18 13:52:10 +08:00
// exclude discount
2020-02-18 14:52:26 +08:00
if ($user->discount && $dayPrice) {
$dayPrice = $dayPrice - ($dayPrice * $user->discount / 100);
2020-02-18 13:52:10 +08:00
}
2020-03-29 23:49:06 +08:00
$result = $day * $dayPrice;
return $result > 0 ? $result : 0;
2020-02-17 00:35:21 +08:00
}
2020-01-11 13:36:52 +08:00
public function save(OrderSave $request)
{
2020-02-11 15:45:30 +08:00
if ($this->isNotCompleteOrderByUserId($request->session()->get('id'))) {
2020-01-02 00:05:54 +08:00
abort(500, '存在未付款订单,请取消后再试');
}
2020-01-02 00:07:30 +08:00
2019-10-29 15:33:36 +08:00
$plan = Plan::find($request->input('plan_id'));
$user = User::find($request->session()->get('id'));
2020-01-11 13:36:52 +08:00
2019-10-29 15:33:36 +08:00
if (!$plan) {
abort(500, '该订阅不存在');
}
2020-01-11 13:36:52 +08:00
2020-02-11 14:03:23 +08:00
if ((!$plan->show && !$plan->renew) || (!$plan->show && $user->plan_id !== $plan->id)) {
2019-10-29 15:33:36 +08:00
abort(500, '该订阅已售罄');
}
2020-02-11 13:31:58 +08:00
if (!$plan->renew && $user->plan_id == $plan->id) {
2019-10-29 15:33:36 +08:00
abort(500, '该订阅无法续费,请更换其他订阅');
}
2019-11-25 18:56:00 +08:00
2019-12-31 15:58:53 +08:00
if ($plan[$request->input('cycle')] === NULL) {
2019-11-25 18:56:00 +08:00
abort(500, '该订阅周期无法进行购买,请选择其他周期');
}
2020-01-01 17:55:16 +08:00
if ($request->input('coupon_code')) {
$coupon = Coupon::where('code', $request->input('coupon_code'))->first();
if (!$coupon) {
abort(500, '优惠券无效');
}
2020-01-01 23:42:25 +08:00
if ($coupon->limit_use <= 0 && $coupon->limit_use !== NULL) {
2020-01-01 23:40:14 +08:00
abort(500, '优惠券已无可用次数');
}
2020-01-01 17:55:16 +08:00
if (time() < $coupon->started_at) {
abort(500, '优惠券还未到可用时间');
}
if (time() > $coupon->ended_at) {
abort(500, '优惠券已过期');
}
}
2020-01-11 13:36:52 +08:00
2020-01-01 23:40:14 +08:00
DB::beginTransaction();
2019-10-29 15:33:36 +08:00
$order = new Order();
$order->user_id = $request->session()->get('id');
$order->plan_id = $plan->id;
$order->cycle = $request->input('cycle');
$order->trade_no = Helper::guid();
$order->total_amount = $plan[$request->input('cycle')];
2020-03-17 20:00:46 +08:00
// coupon start
2020-01-01 17:55:16 +08:00
if (isset($coupon)) {
switch ($coupon->type) {
2020-01-11 13:36:52 +08:00
case 1:
$order->discount_amount = $coupon->value;
2020-01-01 18:05:44 +08:00
break;
2020-01-11 13:36:52 +08:00
case 2:
$order->discount_amount = $order->total_amount * ($coupon->value / 100);
2020-01-01 18:05:44 +08:00
break;
2020-01-01 17:55:16 +08:00
}
2020-01-01 23:48:23 +08:00
if ($coupon->limit_use !== NULL) {
$coupon->limit_use = $coupon->limit_use - 1;
if (!$coupon->save()) {
DB::rollback();
abort(500, '优惠券使用失败');
}
}
2020-01-01 17:55:16 +08:00
}
2020-03-17 20:00:46 +08:00
// coupon complete
// discount start
2020-02-14 21:54:16 +08:00
if ($user->discount) {
2020-02-15 00:55:04 +08:00
$order->discount_amount = $order->discount_amount + ($order->total_amount * ($user->discount / 100));
2020-02-14 21:54:16 +08:00
}
2020-02-15 00:55:04 +08:00
// discount end
2020-03-17 20:00:46 +08:00
$order->total_amount = $order->total_amount - $order->discount_amount;
2020-02-18 12:42:05 +08:00
// renew and change subscribe process
2020-02-29 15:22:01 +08:00
if ($user->plan_id !== NULL && $order->plan_id !== $user->plan_id) {
2020-03-04 21:45:06 +08:00
if (!(int)config('v2board.plan_change_enable', 1)) abort(500, '目前不允许更改订阅,请联系客服或提交工单');
2020-02-18 12:42:05 +08:00
$order->type = 3;
$order->surplus_amount = $this->getSurplusValue($user);
if ($order->surplus_amount >= $order->total_amount) {
$order->refund_amount = $order->surplus_amount - $order->total_amount;
$order->total_amount = 0;
} else {
$order->total_amount = $order->total_amount - $order->surplus_amount;
}
} else if ($user->expired_at > time() && $order->plan_id == $user->plan_id) {
$order->type = 2;
} else {
$order->type = 1;
}
2020-01-12 02:32:15 +08:00
// invite process
if ($user->invite_user_id && $order->total_amount > 0) {
2020-03-15 21:26:56 +08:00
$order->invite_user_id = $user->invite_user_id;
2020-03-15 21:32:12 +08:00
$commissionFirstTime = (int)config('v2board.commission_first_time_enable', 1);
2020-03-15 21:28:57 +08:00
if (!$commissionFirstTime || ($commissionFirstTime && !Order::where('user_id', $user->id)->where('status', 3)->first())) {
$inviter = User::find($user->invite_user_id);
if ($inviter && $inviter->commission_rate) {
$order->commission_balance = $order->total_amount * ($inviter->commission_rate / 100);
} else {
$order->commission_balance = $order->total_amount * (config('v2board.invite_commission', 10) / 100);
}
2020-01-12 02:32:15 +08:00
}
}
2020-03-17 20:00:46 +08:00
// use balance
if ($user->balance && $order->total_amount > 0) {
$remainingBalance = $user->balance - $order->total_amount;
$userService = new UserService();
if ($remainingBalance > 0) {
2020-03-17 20:04:00 +08:00
if (!$userService->addBalance($order->user_id, - $order->total_amount)) {
2020-03-17 20:00:46 +08:00
DB::rollBack();
abort(500, '余额不足');
}
$order->balance_amount = $order->total_amount;
$order->total_amount = 0;
} else {
2020-03-17 20:04:00 +08:00
if (!$userService->addBalance($order->user_id, - $user->balance)) {
2020-03-17 20:00:46 +08:00
DB::rollBack();
abort(500, '余额不足');
}
$order->balance_amount = $user->balance;
$order->total_amount = $order->total_amount - $user->balance;
}
}
2019-10-29 15:33:36 +08:00
if (!$order->save()) {
2020-01-01 23:40:14 +08:00
DB::rollback();
2019-10-29 15:33:36 +08:00
abort(500, '订单创建失败');
}
2020-01-11 13:36:52 +08:00
2020-01-01 23:40:14 +08:00
DB::commit();
2019-10-29 15:33:36 +08:00
return response([
'data' => $order->trade_no
]);
}
2020-01-11 13:36:52 +08:00
public function checkout(Request $request)
{
2019-10-29 15:33:36 +08:00
$tradeNo = $request->input('trade_no');
$method = $request->input('method');
$order = Order::where('trade_no', $tradeNo)
->where('user_id', $request->session()->get('id'))
->where('status', 0)
->first();
if (!$order) {
2019-12-27 01:58:21 +08:00
abort(500, '订单不存在或已支付');
2019-10-29 15:33:36 +08:00
}
2020-02-17 01:27:07 +08:00
// free process
if ($order->total_amount <= 0) {
$order->total_amount = 0;
$order->status = 1;
$order->save();
exit();
}
2019-10-29 15:33:36 +08:00
switch ($method) {
// return type => 0: QRCode / 1: URL
case 0:
// alipayF2F
if (!(int)config('v2board.alipay_enable')) {
abort(500, '支付方式不可用');
}
return response([
'type' => 0,
'data' => $this->alipayF2F($tradeNo, $order->total_amount)
]);
case 2:
// stripeAlipay
if (!(int)config('v2board.stripe_alipay_enable')) {
abort(500, '支付方式不可用');
}
return response([
'type' => 1,
'data' => $this->stripeAlipay($order)
]);
case 3:
// stripeWepay
if (!(int)config('v2board.stripe_wepay_enable')) {
abort(500, '支付方式不可用');
}
return response([
'type' => 0,
'data' => $this->stripeWepay($order)
]);
2019-12-27 01:50:26 +08:00
case 4:
// bitpayX
if (!(int)config('v2board.bitpayx_enable')) {
abort(500, '支付方式不可用');
}
return response([
'type' => 1,
'data' => $this->bitpayX($order)
]);
2020-01-14 00:29:59 +08:00
case 5:
if (!(int)config('v2board.paytaro_enable')) {
abort(500, '支付方式不可用');
}
return response([
'type' => 1,
'data' => $this->payTaro($order)
]);
2019-10-29 15:33:36 +08:00
default:
abort(500, '支付方式不存在');
}
}
2020-01-11 13:36:52 +08:00
public function check(Request $request)
{
2019-10-29 15:33:36 +08:00
$tradeNo = $request->input('trade_no');
$order = Order::where('trade_no', $tradeNo)
->where('user_id', $request->session()->get('id'))
->first();
if (!$order) {
abort(500, '订单不存在');
}
return response([
'data' => $order->status
]);
}
2020-01-11 13:36:52 +08:00
public function getPaymentMethod()
{
2019-10-29 15:33:36 +08:00
$data = [];
if ((int)config('v2board.alipay_enable')) {
$alipayF2F = new \StdClass();
$alipayF2F->name = '支付宝';
$alipayF2F->method = 0;
$alipayF2F->icon = 'alipay';
array_push($data, $alipayF2F);
}
if ((int)config('v2board.stripe_alipay_enable')) {
$stripeAlipay = new \StdClass();
$stripeAlipay->name = '支付宝';
$stripeAlipay->method = 2;
$stripeAlipay->icon = 'alipay';
array_push($data, $stripeAlipay);
}
if ((int)config('v2board.stripe_wepay_enable')) {
$stripeWepay = new \StdClass();
$stripeWepay->name = '微信';
$stripeWepay->method = 3;
$stripeWepay->icon = 'wechat';
array_push($data, $stripeWepay);
}
2019-12-27 01:50:26 +08:00
if ((int)config('v2board.bitpayx_enable')) {
$bitpayX = new \StdClass();
2020-01-17 00:20:54 +08:00
$bitpayX->name = '聚合支付';
2019-12-27 01:50:26 +08:00
$bitpayX->method = 4;
2020-01-20 17:13:14 +08:00
$bitpayX->icon = 'wallet';
2019-12-27 01:50:26 +08:00
array_push($data, $bitpayX);
}
2020-01-14 00:29:59 +08:00
if ((int)config('v2board.paytaro_enable')) {
$obj = new \StdClass();
2020-01-17 00:20:54 +08:00
$obj->name = '聚合支付';
2020-01-14 00:29:59 +08:00
$obj->method = 5;
2020-01-20 17:13:14 +08:00
$obj->icon = 'wallet';
2020-01-14 00:29:59 +08:00
array_push($data, $obj);
}
2019-10-29 15:33:36 +08:00
return response([
'data' => $data
]);
}
2020-01-11 13:36:52 +08:00
public function cancel(Request $request)
{
2020-01-02 00:05:54 +08:00
if (empty($request->input('trade_no'))) {
abort(500, '参数有误');
}
$order = Order::where('trade_no', $request->input('trade_no'))
->where('user_id', $request->session()->get('id'))
->first();
if (!$order) {
abort(500, '订单不存在');
}
2020-01-02 00:21:39 +08:00
if ($order->status !== 0) {
2020-01-02 00:19:04 +08:00
abort(500, '只可以取消待支付订单');
}
2020-03-17 20:16:55 +08:00
$orderService = new OrderService($order);
if (!$orderService->cancel()) {
2020-01-02 00:05:54 +08:00
abort(500, '取消失败');
}
return response([
'data' => true
]);
}
2020-01-11 13:36:52 +08:00
private function alipayF2F($tradeNo, $totalAmount)
{
2019-10-29 15:33:36 +08:00
$gateway = Omnipay::create('Alipay_AopF2F');
$gateway->setSignType('RSA2'); //RSA/RSA2
$gateway->setAppId(config('v2board.alipay_appid'));
$gateway->setPrivateKey(config('v2board.alipay_privkey')); // 可以是路径,也可以是密钥内容
$gateway->setAlipayPublicKey(config('v2board.alipay_pubkey')); // 可以是路径,也可以是密钥内容
2019-12-10 10:52:09 +08:00
$gateway->setNotifyUrl(url('/api/v1/guest/order/alipayNotify'));
2019-10-29 15:33:36 +08:00
$request = $gateway->purchase();
$request->setBizContent([
2020-01-11 13:36:52 +08:00
'subject' => config('v2board.app_name', 'V2Board') . ' - 订阅',
2019-10-29 15:33:36 +08:00
'out_trade_no' => $tradeNo,
'total_amount' => $totalAmount / 100
]);
/** @var \Omnipay\Alipay\Responses\AopTradePreCreateResponse $response */
$response = $request->send();
$result = $response->getAlipayResponse();
if ($result['code'] !== '10000') {
2020-01-11 13:36:52 +08:00
abort(500, $result['sub_msg']);
2019-10-29 15:33:36 +08:00
}
// 获取收款二维码内容
return $response->getQrCode();
}
2020-01-11 13:36:52 +08:00
private function stripeAlipay($order)
{
2020-03-05 00:06:40 +08:00
$currency = config('stripe_currency', 'hkd');
$exchange = Helper::exchange('CNY', strtoupper($currency));
2019-10-29 15:33:36 +08:00
if (!$exchange) {
abort(500, '货币转换超时,请稍后再试');
}
Stripe::setApiKey(config('v2board.stripe_sk_live'));
$source = Source::create([
'amount' => floor($order->total_amount * $exchange),
2020-03-05 00:06:40 +08:00
'currency' => $currency,
2019-10-29 15:33:36 +08:00
'type' => 'alipay',
2020-03-05 23:41:44 +08:00
'statement_descriptor' => $order->trade_no,
'metadata' => [
'user_id' => $order->user_id,
'invoice_id' => $order->trade_no,
'identifier' => ''
],
2019-10-29 15:33:36 +08:00
'redirect' => [
2019-12-21 00:01:11 +08:00
'return_url' => config('v2board.app_url', env('APP_URL')) . '/#/order'
2019-10-29 15:33:36 +08:00
]
]);
if (!$source['redirect']['url']) {
abort(500, '支付网关请求失败');
}
2020-01-11 13:36:52 +08:00
2020-01-12 02:15:48 +08:00
if (!Cache::put($source['id'], $order->trade_no, 3600)) {
2019-11-29 19:52:55 +08:00
abort(500, '订单创建失败');
2019-10-29 15:33:36 +08:00
}
return $source['redirect']['url'];
}
2020-01-11 13:36:52 +08:00
private function stripeWepay($order)
{
2020-03-05 00:06:40 +08:00
$currency = config('stripe_currency', 'hkd');
$exchange = Helper::exchange('CNY', strtoupper($currency));
2019-10-29 15:33:36 +08:00
if (!$exchange) {
abort(500, '货币转换超时,请稍后再试');
}
Stripe::setApiKey(config('v2board.stripe_sk_live'));
$source = Source::create([
'amount' => floor($order->total_amount * $exchange),
2020-03-05 00:06:40 +08:00
'currency' => $currency,
2019-10-29 15:33:36 +08:00
'type' => 'wechat',
2020-03-05 23:41:44 +08:00
'metadata' => [
'user_id' => $order->user_id,
'invoice_id' => $order->trade_no,
'identifier' => ''
2020-03-07 11:37:39 +08:00
],
2019-10-29 15:33:36 +08:00
'redirect' => [
2019-12-21 00:01:11 +08:00
'return_url' => config('v2board.app_url', env('APP_URL')) . '/#/order'
2019-10-29 15:33:36 +08:00
]
]);
if (!$source['wechat']['qr_code_url']) {
abort(500, '支付网关请求失败');
}
2020-01-12 02:16:05 +08:00
if (!Cache::put($source['id'], $order->trade_no, 3600)) {
2019-11-29 19:52:55 +08:00
abort(500, '订单创建失败');
2019-10-29 15:33:36 +08:00
}
return $source['wechat']['qr_code_url'];
}
2019-12-27 01:50:26 +08:00
2020-01-11 13:36:52 +08:00
private function bitpayX($order)
{
2019-12-27 01:50:26 +08:00
$bitpayX = new BitpayX(config('v2board.bitpayx_appsecret'));
2020-01-11 13:36:52 +08:00
$params = [
2020-01-20 23:34:53 +08:00
'merchant_order_id' => $order->trade_no,
2020-01-11 13:36:52 +08:00
'price_amount' => $order->total_amount / 100,
'price_currency' => 'CNY',
'title' => '支付单号:' . $order->trade_no,
'description' => '充值:' . $order->total_amount / 100 . ' 元',
'callback_url' => url('/api/v1/guest/order/bitpayXNotify'),
'success_url' => config('v2board.app_url', env('APP_URL')) . '/#/order',
'cancel_url' => config('v2board.app_url', env('APP_URL')) . '/#/order'
2019-12-27 01:50:26 +08:00
];
$strToSign = $bitpayX->prepareSignId($params['merchant_order_id']);
2020-01-11 13:36:52 +08:00
$params['token'] = $bitpayX->sign($strToSign);
2019-12-27 01:50:26 +08:00
$result = $bitpayX->mprequest($params);
2020-03-07 11:37:39 +08:00
// Log::info('bitpayXSubmit: ' . json_encode($result));
2019-12-27 01:50:26 +08:00
return isset($result['payment_url']) ? $result['payment_url'] : false;
}
2020-01-14 00:29:59 +08:00
private function payTaro($order)
{
$payTaro = new PayTaro(config('v2board.paytaro_app_id'), config('v2board.paytaro_app_secret'));
$result = $payTaro->pay([
'app_id' => config('v2board.paytaro_app_id'),
'out_trade_no' => $order->trade_no,
'total_amount' => $order->total_amount,
'notify_url' => url('/api/v1/guest/order/payTaroNotify'),
2020-01-14 01:16:16 +08:00
'return_url' => config('v2board.app_url', env('APP_URL')) . '/#/order'
2020-01-14 00:29:59 +08:00
]);
return $result;
}
2019-10-29 15:33:36 +08:00
}