1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-11-16 16:13:20 +01:00
invoiceninja/app/PaymentDrivers/Stripe/CreditCard.php

201 lines
7.5 KiB
PHP
Raw Normal View History

2020-06-01 14:03:18 +02:00
<?php
/**
* Invoice Ninja (https://invoiceninja.com).
2020-06-01 14:03:18 +02:00
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
2020-06-01 14:03:18 +02:00
*
2021-06-16 08:58:16 +02:00
* @license https://www.elastic.co/licensing/elastic-license
2020-06-01 14:03:18 +02:00
*/
namespace App\PaymentDrivers\Stripe;
use App\Exceptions\PaymentFailed;
use App\Http\Requests\ClientPortal\Payments\PaymentResponseRequest;
2020-06-01 16:19:03 +02:00
use App\Jobs\Mail\PaymentFailureMailer;
use App\Jobs\Util\SystemLogger;
2020-06-01 14:14:41 +02:00
use App\Models\GatewayType;
use App\Models\Payment;
2020-06-01 16:19:03 +02:00
use App\Models\PaymentType;
use App\Models\SystemLog;
2020-06-01 14:03:18 +02:00
use App\PaymentDrivers\StripePaymentDriver;
use Stripe\PaymentIntent;
use Stripe\PaymentMethod;
2020-06-01 14:03:18 +02:00
class CreditCard
{
public $stripe;
public function __construct(StripePaymentDriver $stripe)
{
$this->stripe = $stripe;
}
public function authorizeView(array $data)
{
$intent['intent'] = $this->stripe->getSetupIntent();
2020-10-20 16:08:21 +02:00
return render('gateways.stripe.credit_card.authorize', array_merge($data, $intent));
2020-06-01 14:03:18 +02:00
}
2020-06-01 14:14:41 +02:00
public function authorizeResponse($request)
{
$this->stripe->init();
2020-06-01 14:14:41 +02:00
$stripe_response = json_decode($request->input('gateway_response'));
2020-06-01 14:14:41 +02:00
$customer = $this->stripe->findOrCreateCustomer();
$this->stripe->attach($stripe_response->payment_method, $customer);
2020-06-01 14:14:41 +02:00
$stripe_method = $this->stripe->getStripePaymentMethod($stripe_response->payment_method);
2020-06-01 14:14:41 +02:00
$this->storePaymentMethod($stripe_method, $request->payment_method_id, $customer);
2020-06-01 14:14:41 +02:00
return redirect()->route('client.payment_methods.index');
}
2020-06-01 14:29:41 +02:00
public function paymentView(array $data)
{
$payment_intent_data = [
'amount' => $this->stripe->convertToStripeAmount($data['total']['amount_with_fee'], $this->stripe->client->currency()->precision),
2020-06-01 14:29:41 +02:00
'currency' => $this->stripe->client->getCurrencyCode(),
'customer' => $this->stripe->findOrCreateCustomer(),
'description' => ctrans('texts.invoices') . ': ' . collect($data['invoices'])->pluck('invoice_number'), // TODO: More meaningful description.
2020-06-01 14:29:41 +02:00
];
$payment_intent_data['setup_future_usage'] = 'off_session';
2020-06-01 14:29:41 +02:00
$data['intent'] = $this->stripe->createPaymentIntent($payment_intent_data);
$data['gateway'] = $this->stripe;
2020-10-20 16:08:21 +02:00
return render('gateways.stripe.credit_card.pay', $data);
2020-06-01 14:29:41 +02:00
}
public function paymentResponse(PaymentResponseRequest $request)
2020-06-01 14:29:41 +02:00
{
$this->stripe->init();
2020-08-30 14:00:19 +02:00
2020-06-01 16:19:03 +02:00
$state = [
'server_response' => json_decode($request->gateway_response),
'payment_hash' => $request->payment_hash,
2020-06-01 16:19:03 +02:00
];
$state = array_merge($state, $request->all());
$state['store_card'] = boolval($state['store_card']);
2020-06-01 16:19:03 +02:00
2021-02-03 16:49:29 +01:00
if ($request->has('token') && !is_null($request->token)) {
$state['store_card'] = false;
}
2021-05-12 00:14:29 +02:00
$state['payment_intent'] = PaymentIntent::retrieve($state['server_response']->id, $this->stripe->stripe_connect_auth);
$state['customer'] = $state['payment_intent']->customer;
2020-06-01 16:19:03 +02:00
$this->stripe->payment_hash->data = array_merge((array) $this->stripe->payment_hash->data, $state);
$this->stripe->payment_hash->save();
2020-06-01 16:19:03 +02:00
$server_response = $this->stripe->payment_hash->data->server_response;
2020-08-30 14:00:19 +02:00
if ($server_response->status == 'succeeded') {
2020-08-30 14:00:19 +02:00
2021-01-11 23:14:52 +01:00
$this->stripe->logSuccessfulGatewayResponse(['response' => json_decode($request->gateway_response), 'data' => $this->stripe->payment_hash], SystemLog::TYPE_STRIPE);
2021-01-11 23:06:56 +01:00
return $this->processSuccessfulPayment();
2020-06-01 16:19:03 +02:00
}
return $this->processUnsuccessfulPayment($server_response);
}
private function processSuccessfulPayment()
2020-06-01 16:19:03 +02:00
{
$stripe_method = $this->stripe->getStripePaymentMethod($this->stripe->payment_hash->data->server_response->payment_method);
$data = [
'payment_method' => $this->stripe->payment_hash->data->server_response->payment_method,
'payment_type' => PaymentType::parseCardType(strtolower($stripe_method->card->brand)),
'amount' => $this->stripe->convertFromStripeAmount($this->stripe->payment_hash->data->server_response->amount, $this->stripe->client->currency()->precision),
'transaction_reference' => optional($this->stripe->payment_hash->data->payment_intent->charges->data[0])->id,
2021-01-27 11:38:28 +01:00
'gateway_type_id' => GatewayType::CREDIT_CARD,
];
$this->stripe->payment_hash->data = array_merge((array) $this->stripe->payment_hash->data, ['amount' => $data['amount']]);
$this->stripe->payment_hash->save();
if ($this->stripe->payment_hash->data->store_card) {
2020-11-16 14:10:45 +01:00
$customer = new \stdClass;
$customer->id = $this->stripe->payment_hash->data->customer;
2020-06-01 16:19:03 +02:00
$this->stripe->attach($this->stripe->payment_hash->data->server_response->payment_method, $customer);
2020-11-16 14:10:45 +01:00
$stripe_method = $this->stripe->getStripePaymentMethod($this->stripe->payment_hash->data->server_response->payment_method);
$this->storePaymentMethod($stripe_method, $this->stripe->payment_hash->data->payment_method_id, $customer);
2020-06-01 16:19:03 +02:00
}
$payment = $this->stripe->createPayment($data, Payment::STATUS_COMPLETED);
2020-06-01 16:19:03 +02:00
SystemLogger::dispatch(
['response' => $this->stripe->payment_hash->data->server_response, 'data' => $data],
SystemLog::CATEGORY_GATEWAY_RESPONSE,
SystemLog::EVENT_GATEWAY_SUCCESS,
SystemLog::TYPE_STRIPE,
$this->stripe->client,
$this->stripe->client->company,
);
2020-06-01 16:19:03 +02:00
return redirect()->route('client.payments.show', ['payment' => $this->stripe->encodePrimaryKey($payment->id)]);
}
private function processUnsuccessfulPayment($server_response)
{
PaymentFailureMailer::dispatch($this->stripe->client, $server_response->cancellation_reason, $this->stripe->client->company, $server_response->amount);
PaymentFailureMailer::dispatch(
$this->stripe->client,
$server_response,
$this->stripe->client->company,
$server_response->amount
);
2020-06-01 16:19:03 +02:00
$message = [
'server_response' => $server_response,
'data' => $this->stripe->payment_hash->data,
2020-06-01 16:19:03 +02:00
];
SystemLogger::dispatch(
$message,
SystemLog::CATEGORY_GATEWAY_RESPONSE,
SystemLog::EVENT_GATEWAY_FAILURE,
SystemLog::TYPE_STRIPE,
$this->stripe->client,
$this->stripe->client->company,
);
2020-06-01 16:19:03 +02:00
throw new PaymentFailed('Failed to process the payment.', 500);
2020-06-01 16:19:03 +02:00
}
private function storePaymentMethod(PaymentMethod $method, $payment_method_id, $customer)
{
try {
$payment_meta = new \stdClass;
$payment_meta->exp_month = (string) $method->card->exp_month;
$payment_meta->exp_year = (string) $method->card->exp_year;
$payment_meta->brand = (string) $method->card->brand;
$payment_meta->last4 = (string) $method->card->last4;
$payment_meta->type = GatewayType::CREDIT_CARD;
$data = [
'payment_meta' => $payment_meta,
'token' => $method->id,
'payment_method_id' => $payment_method_id,
];
$this->stripe->storeGatewayToken($data, ['gateway_customer_reference' => $customer->id]);
} catch (\Exception $e) {
return $this->stripe->processInternallyFailedPayment($this->stripe, $e);
}
}
2020-06-01 14:14:41 +02:00
}