| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181 |
- <?php
- namespace App\Payments;
- use App\Models\Payment;
- use App\Payments\Library\Gateway;
- use Auth;
- use Exception;
- use Http;
- use Illuminate\Http\JsonResponse;
- use Log;
- use Response;
- use Stripe\Checkout\Session;
- use Stripe\Exception\SignatureVerificationException;
- use Stripe\Source;
- use Stripe\Webhook;
- use UnexpectedValueException;
- class Stripe extends Gateway
- {
- public function __construct()
- {
- \Stripe\Stripe::setApiKey(sysConfig('stripe_secret_key'));
- }
- public function purchase($request): JsonResponse
- {
- $type = $request->input('type');
- $payment = $this->creatNewPayment(Auth::id(), $request->input('id'), $request->input('amount'));
- if ($type == 1 || $type == 3) {
- $stripe_currency = sysConfig('stripe_currency');
- $response = Http::get('https://api.exchangerate-api.com/v4/latest/'.strtoupper($stripe_currency));
- if (! $response->ok()) {
- Log::warning('【Stripe】错误: 获取汇率失败!');
- $payment->delete();
- return Response::json(['status' => 'fail', 'message' => '获取汇率失败!']);
- }
- $response = $response->json();
- $price_exchanged = bcdiv((float) $payment->amount, $response['rates']['CNY'], 10);
- $source = Source::create([
- 'amount' => floor($price_exchanged * 100),
- 'currency' => $stripe_currency,
- 'type' => $type == 1 ? 'alipay' : 'wechat',
- 'statement_descriptor' => $payment->trade_no,
- 'metadata' => [
- 'user_id' => $payment->user_id,
- 'out_trade_no' => $payment->trade_no,
- 'identifier' => '',
- ],
- 'redirect' => [
- 'return_url' => route('invoice'),
- ],
- ]);
- if ($type == 3) {
- if (! $source['wechat']['qr_code_url']) {
- Log::warning('创建订单错误:未知错误');
- $payment->delete();
- return Response::json(['status' => 'fail', 'message' => '创建订单失败:未知错误']);
- }
- $payment->update(['qr_code' => 1, 'url' => $source['wechat']['qr_code_url']]);
- return Response::json(['status' => 'success', 'data' => $payment->trade_no, 'message' => '创建订单成功!']);
- } else {
- if (! $source['redirect']['url']) {
- Log::warning('创建订单错误:未知错误');
- $payment->delete();
- return response()->json(['code' => 0, 'msg' => '创建订单失败:未知错误']);
- }
- $payment->update(['url' => $source['redirect']['url']]);
- return Response::json(['status' => 'success', 'url' => $source['redirect']['url'], 'message' => '创建订单成功!']);
- }
- } else {
- $data = $this->getCheckoutSessionData($payment->trade_no, $payment->amount, $type);
- try {
- $session = Session::create($data);
- $url = route('stripe.checkout', ['session_id' => $session->id]);
- $payment->update(['url' => $url]);
- return Response::json(['status' => 'success', 'url' => $url, 'message' => '创建订单成功!']);
- } catch (Exception $e) {
- Log::error('【Stripe】错误: '.$e->getMessage());
- exit;
- }
- }
- }
- public function redirectPage($session_id)
- { // redirect to Stripe Payment url
- return view('user.components.payment.stripe', ['session_id' => $session_id]);
- }
- public function notify($request): void
- { // url = '/callback/notify?method=stripe'
- $sigHeader = $_SERVER['HTTP_STRIPE_SIGNATURE'];
- $endpointSecret = sysConfig('stripe_signing_secret');
- $payload = @file_get_contents('php://input');
- try {
- $event = Webhook::constructEvent($payload, $sigHeader, $endpointSecret);
- } catch (UnexpectedValueException $e) {
- // Invalid payload
- http_response_code(400);
- exit();
- } catch (SignatureVerificationException $e) {
- // Invalid signature
- http_response_code(400);
- exit();
- }
- Log::info('【Stripe】Passed signature verification!');
- switch ($event->type) {
- case 'checkout.session.completed':
- /* @var $session Session */
- $session = $event->data->object;
- // Check if the order is paid (e.g., from a card payment)
- //
- // A delayed notification payment will have an `unpaid` status, as
- // you're still waiting for funds to be transferred from the customer's
- // account.
- if ($session->payment_status == 'paid') {
- // Fulfill the purchase
- $this->paymentReceived($session->client_reference_id);
- }
- break;
- case 'checkout.session.async_payment_succeeded':
- $session = $event->data->object;
- // Fulfill the purchase
- $this->paymentReceived($session->client_reference_id);
- break;
- case 'checkout.session.async_payment_failed':
- $session = $event->data->object;
- // Send an email to the customer asking them to retry their order
- $this->failedPayment($session);
- break;
- }
- http_response_code(200);
- exit();
- }
- public function failedPayment(Session $session)
- { // 未支付成功则关闭订单
- $payment = Payment::whereTradeNo($session->client_reference_id)->first();
- if ($payment) {
- $payment->order->close();
- }
- }
- protected function getCheckoutSessionData(string $tradeNo, int $amount, int $type): array
- {
- $unitAmount = $amount * 100;
- return [
- 'payment_method_types' => ['card'],
- 'line_items' => [
- [
- 'price_data' => [
- 'currency' => 'usd',
- 'product_data' => ['name' => sysConfig('subject_name') ?: sysConfig('website_name')],
- 'unit_amount' => $unitAmount,
- ],
- 'quantity' => 1,
- ],
- ],
- 'mode' => 'payment',
- 'success_url' => route('invoice'),
- 'cancel_url' => route('invoice'),
- 'client_reference_id' => $tradeNo,
- 'customer_email' => Auth::getUser()->email,
- ];
- }
- }
|