1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-11-15 07:33:04 +01:00
invoiceninja/app/PaymentDrivers/Stripe/ACH.php

652 lines
24 KiB
PHP
Raw Normal View History

2020-06-09 14:40:55 +02:00
<?php
/**
* Invoice Ninja (https://invoiceninja.com).
2020-06-09 14:40:55 +02:00
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
2024-04-12 06:15:41 +02:00
* @copyright Copyright (c) 2024. Invoice Ninja LLC (https://invoiceninja.com)
2020-06-09 14:40:55 +02:00
*
2021-06-16 08:58:16 +02:00
* @license https://www.elastic.co/licensing/elastic-license
2020-06-09 14:40:55 +02:00
*/
namespace App\PaymentDrivers\Stripe;
use App\Exceptions\PaymentFailed;
use App\Http\Requests\Request;
use App\Jobs\Mail\NinjaMailerJob;
use App\Jobs\Mail\NinjaMailerObject;
2020-06-09 14:40:55 +02:00
use App\Jobs\Util\SystemLogger;
use App\Mail\Gateways\ACHVerificationNotification;
2020-06-09 14:40:55 +02:00
use App\Models\ClientGatewayToken;
use App\Models\GatewayType;
use App\Models\Payment;
use App\Models\PaymentHash;
2020-06-09 14:40:55 +02:00
use App\Models\PaymentType;
use App\Models\SystemLog;
use App\PaymentDrivers\Common\LivewireMethodInterface;
2020-06-09 14:40:55 +02:00
use App\PaymentDrivers\StripePaymentDriver;
use App\Utils\Traits\MakesHash;
use Exception;
use Stripe\Customer;
2022-05-19 07:48:31 +02:00
use Stripe\Exception\ApiErrorException;
use Stripe\Exception\AuthenticationException;
2020-11-02 16:29:02 +01:00
use Stripe\Exception\CardException;
2020-06-09 14:40:55 +02:00
use Stripe\Exception\InvalidRequestException;
2022-05-19 07:48:31 +02:00
use Stripe\Exception\RateLimitException;
use Stripe\PaymentIntent;
2020-06-09 14:40:55 +02:00
class ACH implements LivewireMethodInterface
2020-06-09 14:40:55 +02:00
{
use MakesHash;
2020-06-09 14:40:55 +02:00
/** @var StripePaymentDriver */
public $stripe;
public function __construct(StripePaymentDriver $stripe)
{
$this->stripe = $stripe;
}
2022-05-19 07:48:31 +02:00
/**
* Authorize a bank account - requires microdeposit verification
*/
2020-06-09 14:40:55 +02:00
public function authorizeView(array $data)
{
$data['gateway'] = $this->stripe;
2020-06-09 14:40:55 +02:00
return render('gateways.stripe.ach.authorize', array_merge($data));
}
public function authorizeResponse(Request $request)
2020-06-09 14:40:55 +02:00
{
$this->stripe->init();
$stripe_response = json_decode($request->input('gateway_response'));
$customer = $this->stripe->findOrCreateCustomer();
2020-06-09 14:40:55 +02:00
try {
2023-02-16 02:36:09 +01:00
$source = Customer::createSource($customer->id, ['source' => $stripe_response->token->id], array_merge($this->stripe->stripe_connect_auth, ['idempotency_key' => uniqid("st", true)]));
2020-06-09 14:40:55 +02:00
} catch (InvalidRequestException $e) {
throw new PaymentFailed($e->getMessage(), $e->getCode());
2020-06-09 14:40:55 +02:00
}
$client_gateway_token = $this->storePaymentMethod($source, $request->input('method'), $customer);
2020-06-09 14:40:55 +02:00
2021-08-10 10:13:04 +02:00
$verification = route('client.payment_methods.verification', ['payment_method' => $client_gateway_token->hashed_id, 'method' => GatewayType::BANK_TRANSFER], false);
$mailer = new NinjaMailerObject();
2021-08-10 10:13:04 +02:00
$mailer->mailable = new ACHVerificationNotification(
auth()->guard('contact')->user()->client->company,
route('client.contact_login', ['contact_key' => auth()->guard('contact')->user()->contact_key, 'next' => $verification])
2021-08-10 10:13:04 +02:00
);
$mailer->company = auth()->guard('contact')->user()->client->company;
$mailer->settings = auth()->guard('contact')->user()->client->company->settings;
$mailer->to_user = auth()->guard('contact')->user();
NinjaMailerJob::dispatch($mailer);
2020-10-20 17:54:08 +02:00
return redirect()->route('client.payment_methods.verification', ['payment_method' => $client_gateway_token->hashed_id, 'method' => GatewayType::BANK_TRANSFER]);
2020-06-09 14:40:55 +02:00
}
public function updateBankAccount(array $event)
{
$stripe_event = $event['data']['object'];
2023-01-11 01:37:11 +01:00
2023-08-07 00:23:13 +02:00
$token = ClientGatewayToken::query()->where('token', $stripe_event['id'])
->where('gateway_customer_reference', $stripe_event['customer'])
->first();
2023-02-16 02:36:09 +01:00
if ($token && isset($stripe_event['object']) && $stripe_event['object'] == 'bank_account' && isset($stripe_event['status']) && $stripe_event['status'] == 'verified') {
$meta = $token->meta;
$meta->state = 'authorized';
$token->meta = $meta;
$token->save();
}
}
2020-06-09 14:40:55 +02:00
public function verificationView(ClientGatewayToken $token)
{
//double check here if we need to show the verification view.
$this->stripe->init();
2024-06-14 09:09:44 +02:00
if(substr($token->token, 0, 2) == 'pm') {
2024-03-19 22:43:49 +01:00
$pm = $this->stripe->getStripePaymentMethod($token->token);
2024-06-14 09:09:44 +02:00
if(!$pm->customer) {
2024-03-19 22:43:49 +01:00
$meta = $token->meta;
$meta->state = 'unauthorized';
$token->meta = $meta;
$token->save();
return redirect()
->route('client.payment_methods.show', $token->hashed_id);
}
if (isset($token->meta->state) && $token->meta->state === 'authorized') {
return redirect()
->route('client.payment_methods.show', $token->hashed_id)
->with('message', __('texts.payment_method_verified'));
}
2024-06-14 09:09:44 +02:00
if($token->meta->next_action) {
2024-03-19 22:43:49 +01:00
return redirect($token->meta->next_action);
2024-06-14 09:09:44 +02:00
}
2024-03-19 22:43:49 +01:00
}
$bank_account = Customer::retrieveSource($token->gateway_customer_reference, $token->token, [], $this->stripe->stripe_connect_auth);
/* Catch externally validated bank accounts and mark them as verified */
2023-02-16 02:36:09 +01:00
if (isset($bank_account->status) && $bank_account->status == 'verified') {
$meta = $token->meta;
$meta->state = 'authorized';
$token->meta = $meta;
$token->save();
return redirect()
->route('client.payment_methods.show', $token->hashed_id)
->with('message', __('texts.payment_method_verified'));
}
$data = [
'token' => $token,
'gateway' => $this->stripe,
];
return render('gateways.stripe.ach.verify', $data);
2020-06-09 14:40:55 +02:00
}
public function processVerification(Request $request, ClientGatewayToken $token)
2020-06-09 14:40:55 +02:00
{
$request->validate([
'transactions.*' => ['integer', 'min:1'],
]);
if (isset($token->meta->state) && $token->meta->state === 'authorized') {
return redirect()
->route('client.payment_methods.show', $token->hashed_id)
->with('message', __('texts.payment_method_verified'));
}
2020-06-09 14:40:55 +02:00
$this->stripe->init();
2021-08-04 03:31:06 +02:00
$bank_account = Customer::retrieveSource($request->customer, $request->source, [], $this->stripe->stripe_connect_auth);
2020-06-09 14:40:55 +02:00
try {
$bank_account->verify(['amounts' => request()->transactions]);
2020-06-09 14:40:55 +02:00
$meta = $token->meta;
$meta->state = 'authorized';
$token->meta = $meta;
2020-06-09 14:40:55 +02:00
$token->save();
return redirect()
->route('client.payment_methods.show', $token->hashed_id)
->with('message', __('texts.payment_method_verified'));
2020-10-28 11:10:49 +01:00
} catch (CardException $e) {
2020-06-09 14:40:55 +02:00
return back()->with('error', $e->getMessage());
}
}
2022-05-19 07:48:31 +02:00
/**
* Make a payment WITH instant verification.
*/
2020-06-09 14:40:55 +02:00
public function paymentView(array $data)
{
$data = $this->paymentData($data);
2022-05-18 01:39:54 +02:00
2024-10-08 01:11:43 +02:00
if(!$data['authorized']){
$token = $data['tokens'][0];
return redirect()->route('client.payment_methods.show', $token->hashed_id);
}
2022-05-18 04:59:24 +02:00
return render('gateways.stripe.ach.pay', $data);
2020-06-09 14:40:55 +02:00
}
public function tokenBilling(ClientGatewayToken $cgt, PaymentHash $payment_hash)
{
$amount = array_sum(array_column($payment_hash->invoices(), 'amount')) + $payment_hash->fee_total;
$description = $this->stripe->getDescription(false);
if (substr($cgt->token, 0, 2) === 'pm') {
return $this->paymentIntentTokenBilling($amount, $description, $cgt, false);
}
2022-05-19 07:48:31 +02:00
$this->stripe->init();
$response = null;
try {
$state = [
'gateway_type_id' => GatewayType::BANK_TRANSFER,
'amount' => $this->stripe->convertToStripeAmount($amount, $this->stripe->client->currency()->precision, $this->stripe->client->currency()),
'currency' => $this->stripe->client->getCurrencyCode(),
'customer' => $cgt->gateway_customer_reference,
'source' => $cgt->token,
];
$state['charge'] = \Stripe\Charge::create([
'amount' => $state['amount'],
'currency' => $state['currency'],
'customer' => $state['customer'],
'source' => $state['source'],
'description' => $description,
], $this->stripe->stripe_connect_auth);
$payment_hash->data = array_merge((array) $payment_hash->data, $state);
$payment_hash->save();
if ($state['charge']->status === 'pending' && is_null($state['charge']->failure_message)) {
return $this->processPendingPayment($state, false);
}
return $this->processUnsuccessfulPayment($state);
} catch (Exception $e) {
if ($e instanceof CardException) {
2021-10-17 13:21:56 +02:00
return redirect()->route('client.payment_methods.verification', ['payment_method' => $cgt->hashed_id, 'method' => GatewayType::BANK_TRANSFER]);
}
throw new PaymentFailed($e->getMessage(), $e->getCode());
}
}
2020-11-01 19:21:31 +01:00
public function paymentIntentTokenBilling($amount, $description, $cgt, $client_present = true)
2022-05-19 07:48:31 +02:00
{
$this->stripe->init();
$response = false;
2022-05-19 07:48:31 +02:00
try {
$data = [
'amount' => $this->stripe->convertToStripeAmount($amount, $this->stripe->client->currency()->precision, $this->stripe->client->currency()),
'currency' => $this->stripe->client->getCurrencyCode(),
'payment_method' => $cgt->token,
'customer' => $cgt->gateway_customer_reference,
'confirm' => true,
'description' => $description,
'metadata' => [
'payment_hash' => $this->stripe->payment_hash->hash,
'gateway_type_id' => $cgt->gateway_type_id,
2022-05-19 07:48:31 +02:00
],
'statement_descriptor' => $this->stripe->getStatementDescriptor(),
2022-05-19 07:48:31 +02:00
];
if ($cgt->gateway_type_id == GatewayType::BANK_TRANSFER) {
2022-05-19 07:48:31 +02:00
$data['payment_method_types'] = ['us_bank_account'];
}
2022-05-19 07:48:31 +02:00
2023-03-11 09:14:04 +01:00
$response = $this->stripe->createPaymentIntent($data);
2022-05-19 07:48:31 +02:00
SystemLogger::dispatch($response, SystemLog::CATEGORY_GATEWAY_RESPONSE, SystemLog::EVENT_GATEWAY_SUCCESS, SystemLog::TYPE_STRIPE, $this->stripe->client, $this->stripe->client->company);
} catch (\Exception $e) {
$data = [
2022-05-19 07:48:31 +02:00
'status' => '',
'error_type' => '',
'error_code' => '',
'param' => '',
'message' => '',
];
switch ($e) {
case $e instanceof CardException:
2023-09-24 05:07:32 +02:00
/** @var CardException $e */
2022-05-19 07:48:31 +02:00
$data['status'] = $e->getHttpStatus();
$data['error_type'] = $e->getError()->type;
$data['error_code'] = $e->getError()->code;
$data['param'] = $e->getError()->param;
$data['message'] = $e->getError()->message;
2023-02-16 02:36:09 +01:00
break;
case $e instanceof RateLimitException:
2022-05-19 07:48:31 +02:00
$data['message'] = 'Too many requests made to the API too quickly';
2023-02-16 02:36:09 +01:00
break;
case $e instanceof InvalidRequestException:
2024-06-14 09:09:44 +02:00
2024-03-19 22:43:49 +01:00
return redirect()->route('client.payment_methods.verification', ['payment_method' => $cgt->hashed_id, 'method' => GatewayType::BANK_TRANSFER]);
case $e instanceof AuthenticationException:
2022-05-19 07:48:31 +02:00
$data['message'] = 'Authentication with Stripe\'s API failed';
2023-02-16 02:36:09 +01:00
break;
case $e instanceof ApiErrorException:
2022-05-19 07:48:31 +02:00
$data['message'] = 'Network communication with Stripe failed';
2023-02-16 02:36:09 +01:00
break;
2022-05-19 07:48:31 +02:00
default:
$data['message'] = $e->getMessage();
2023-02-16 02:36:09 +01:00
break;
2022-05-19 07:48:31 +02:00
}
$this->stripe->processInternallyFailedPayment($this->stripe, $e);
SystemLogger::dispatch($data, SystemLog::CATEGORY_GATEWAY_RESPONSE, SystemLog::EVENT_GATEWAY_FAILURE, SystemLog::TYPE_STRIPE, $this->stripe->client, $this->stripe->client->company);
}
2022-05-19 07:48:31 +02:00
if (! $response) {
return false;
}
$payment_method_type = PaymentType::ACH;
$data = [
'gateway_type_id' => $cgt->gateway_type_id,
'payment_type' => PaymentType::ACH,
2022-11-29 10:09:54 +01:00
'transaction_reference' => isset($response->latest_charge) ? $response->latest_charge : $response->charges->data[0]->id,
2022-05-19 07:48:31 +02:00
'amount' => $amount,
];
$payment = $this->stripe->createPayment($data, Payment::STATUS_PENDING);
$payment->meta = $cgt->meta;
$payment->save();
$this->stripe->payment_hash->payment_id = $payment->id;
$this->stripe->payment_hash->save();
if ($client_present) {
2022-05-19 07:48:31 +02:00
return redirect()->route('client.payments.show', ['payment' => $this->stripe->encodePrimaryKey($payment->id)]);
}
return $payment;
}
2022-05-18 04:59:24 +02:00
public function handlePaymentIntentResponse($request)
{
2022-05-19 07:48:31 +02:00
$response = json_decode($request->gateway_response);
$bank_account_response = json_decode($request->bank_account_response);
2023-03-18 08:24:56 +01:00
if ($response->status == 'requires_source_action' && $response->next_action->type == 'verify_with_microdeposits') {
2023-03-09 07:33:10 +01:00
$method = $bank_account_response->payment_method->us_bank_account;
$method = $bank_account_response->payment_method->us_bank_account;
$method->id = $response->payment_method;
$method->state = 'unauthorized';
$method->next_action = $response->next_action->verify_with_microdeposits->hosted_verification_url;
$customer = $this->stripe->getCustomer($request->customer);
$cgt = $this->storePaymentMethod($method, GatewayType::BANK_TRANSFER, $customer);
return redirect()->route('client.payment_methods.show', ['payment_method' => $cgt->hashed_id]);
}
2022-05-19 07:48:31 +02:00
$method = $bank_account_response->payment_method->us_bank_account;
$method->id = $response->payment_method;
$method->state = 'authorized';
$this->stripe->payment_hash = PaymentHash::where('hash', $request->input('payment_hash'))->first();
2022-05-19 07:48:31 +02:00
if ($response->id && $response->status === 'processing') {
2022-05-19 07:48:31 +02:00
$payment_intent = PaymentIntent::retrieve($response->id, $this->stripe->stripe_connect_auth);
$state = [
'gateway_type_id' => GatewayType::BANK_TRANSFER,
'amount' => $response->amount,
'currency' => $response->currency,
'customer' => $request->customer,
'source' => $response->payment_method,
'charge' => $response,
2022-05-19 07:48:31 +02:00
];
$this->stripe->payment_hash->data = array_merge((array) $this->stripe->payment_hash->data, $state);
2022-05-19 07:48:31 +02:00
$this->stripe->payment_hash->save();
2022-05-19 07:48:31 +02:00
$customer = $this->stripe->getCustomer($request->customer);
$this->storePaymentMethod($method, GatewayType::BANK_TRANSFER, $customer);
return $this->processPendingPayment($state, true);
}
if ($response->next_action) {
2022-05-19 07:48:31 +02:00
}
}
public function processPendingPaymentIntent($state, $client_present = true)
{
$this->stripe->init();
$data = [
'payment_method' => $state['source'],
'payment_type' => PaymentType::ACH,
'amount' => $state['amount'],
'transaction_reference' => $state['charge'],
'gateway_type_id' => GatewayType::BANK_TRANSFER,
];
$payment = $this->stripe->createPayment($data, Payment::STATUS_PENDING);
SystemLogger::dispatch(
['response' => $state, 'data' => $data],
SystemLog::CATEGORY_GATEWAY_RESPONSE,
SystemLog::EVENT_GATEWAY_SUCCESS,
SystemLog::TYPE_STRIPE,
$this->stripe->client,
$this->stripe->client->company,
);
if (! $client_present) {
2022-05-19 07:48:31 +02:00
return $payment;
}
2022-05-19 07:48:31 +02:00
return redirect()->route('client.payments.show', ['payment' => $this->stripe->encodePrimaryKey($payment->id)]);
2022-05-18 04:59:24 +02:00
}
2020-06-09 14:40:55 +02:00
public function paymentResponse($request)
{
2020-11-01 19:21:31 +01:00
$this->stripe->init();
2022-05-18 04:59:24 +02:00
//it may be a payment intent here.
if ($request->input('client_secret') != '') {
2022-05-19 07:48:31 +02:00
return $this->handlePaymentIntentResponse($request);
}
2022-05-18 04:59:24 +02:00
$source = ClientGatewayToken::query()
->where('id', $this->decodePrimaryKey($request->source))
->where('company_id', auth()->guard('contact')->user()->client->company->id)
->first();
if (! $source) {
throw new PaymentFailed(ctrans('texts.payment_token_not_found'), 401);
}
2020-06-09 14:40:55 +02:00
$state = [
'payment_method' => $request->payment_method_id,
'gateway_type_id' => $request->company_gateway_id,
'amount' => $this->stripe->convertToStripeAmount($request->amount, $this->stripe->client->currency()->precision, $this->stripe->client->currency()),
2020-06-09 14:40:55 +02:00
'currency' => $request->currency,
'customer' => $request->customer,
];
2020-11-01 19:21:31 +01:00
$state = array_merge($state, $request->all());
$state['source'] = $source->token;
2020-06-09 14:40:55 +02:00
$this->stripe->payment_hash->data = array_merge((array) $this->stripe->payment_hash->data, $state);
2020-11-01 19:21:31 +01:00
$this->stripe->payment_hash->save();
2020-06-09 14:40:55 +02:00
2021-11-23 18:27:45 +01:00
$amount = array_sum(array_column($this->stripe->payment_hash->invoices(), 'amount')) + $this->stripe->payment_hash->fee_total;
$description = $this->stripe->getDescription(false);
2021-11-23 18:27:45 +01:00
if (substr($source->token, 0, 2) === 'pm') {
return $this->paymentIntentTokenBilling($amount, $description, $source);
}
2022-05-19 07:48:31 +02:00
2020-06-09 14:40:55 +02:00
try {
$state['charge'] = \Stripe\Charge::create([
'amount' => $state['amount'],
'currency' => $state['currency'],
'customer' => $state['customer'],
'source' => $state['source'],
2021-11-23 18:27:45 +01:00
'description' => $description,
2021-04-20 13:30:52 +02:00
], $this->stripe->stripe_connect_auth);
2020-06-09 14:40:55 +02:00
2020-11-01 19:21:31 +01:00
$state = array_merge($state, $request->all());
$this->stripe->payment_hash->data = array_merge((array) $this->stripe->payment_hash->data, $state);
2020-11-01 19:21:31 +01:00
$this->stripe->payment_hash->save();
2020-06-09 14:40:55 +02:00
if ($state['charge']->status === 'pending' && is_null($state['charge']->failure_message)) {
return $this->processPendingPayment($state);
}
return $this->processUnsuccessfulPayment($state);
2020-10-28 11:10:49 +01:00
} catch (Exception $e) {
if ($e instanceof CardException) {
2021-08-04 04:20:38 +02:00
return redirect()->route('client.payment_methods.verification', ['payment_method' => $source->hashed_id, 'method' => GatewayType::BANK_TRANSFER]);
2020-06-09 14:40:55 +02:00
}
throw new PaymentFailed($e->getMessage(), $e->getCode());
2020-06-09 14:40:55 +02:00
}
}
public function processPendingPayment($state, $client_present = true)
2020-06-09 14:40:55 +02:00
{
$this->stripe->init();
$data = [
2020-11-01 19:21:31 +01:00
'payment_method' => $state['source'],
'payment_type' => PaymentType::ACH,
'amount' => $this->stripe->convertFromStripeAmount($this->stripe->payment_hash->data->amount, $this->stripe->client->currency()->precision, $this->stripe->client->currency()),
2020-11-01 19:21:31 +01:00
'transaction_reference' => $state['charge']->id,
2021-01-27 11:38:28 +01:00
'gateway_type_id' => GatewayType::BANK_TRANSFER,
2020-06-09 14:40:55 +02:00
];
$payment = $this->stripe->createPayment($data, Payment::STATUS_PENDING);
2020-11-01 19:21:31 +01:00
SystemLogger::dispatch(
['response' => $state['charge'], 'data' => $data],
SystemLog::CATEGORY_GATEWAY_RESPONSE,
SystemLog::EVENT_GATEWAY_SUCCESS,
SystemLog::TYPE_STRIPE,
2021-05-29 12:57:39 +02:00
$this->stripe->client,
$this->stripe->client->company,
2020-11-01 19:21:31 +01:00
);
2020-06-09 14:40:55 +02:00
if (! $client_present) {
return $payment;
}
2020-06-09 14:40:55 +02:00
return redirect()->route('client.payments.show', ['payment' => $this->stripe->encodePrimaryKey($payment->id)]);
}
public function processUnsuccessfulPayment($state)
{
2021-10-17 12:40:40 +02:00
$this->stripe->sendFailureMail($state['charge']);
2020-11-01 19:21:31 +01:00
2020-06-09 14:40:55 +02:00
$message = [
'server_response' => $state['charge'],
2020-11-01 19:21:31 +01:00
'data' => $this->stripe->payment_hash->data,
2020-06-09 14:40:55 +02:00
];
2020-11-01 19:21:31 +01:00
SystemLogger::dispatch(
$message,
SystemLog::CATEGORY_GATEWAY_RESPONSE,
SystemLog::EVENT_GATEWAY_FAILURE,
SystemLog::TYPE_STRIPE,
$this->stripe->client,
$this->stripe->client->company,
2020-11-01 19:21:31 +01:00
);
2020-06-09 14:40:55 +02:00
2020-11-01 19:21:31 +01:00
throw new PaymentFailed('Failed to process the payment.', 500);
2020-06-09 14:40:55 +02:00
}
private function storePaymentMethod($method, $payment_method_id, $customer)
{
2022-05-19 07:48:31 +02:00
$state = property_exists($method, 'state') ? $method->state : 'unauthorized';
try {
2024-01-14 05:05:00 +01:00
$payment_meta = new \stdClass();
2021-08-09 13:49:39 +02:00
$payment_meta->brand = (string) \sprintf('%s (%s)', $method->bank_name, ctrans('texts.ach'));
$payment_meta->last4 = (string) $method->last4;
$payment_meta->type = GatewayType::BANK_TRANSFER;
2022-05-19 07:48:31 +02:00
$payment_meta->state = $state;
2023-03-18 08:24:56 +01:00
if (property_exists($method, 'next_action')) {
2023-03-09 07:33:10 +01:00
$payment_meta->next_action = $method->next_action;
}
$data = [
'payment_meta' => $payment_meta,
'token' => $method->id,
'payment_method_id' => $payment_method_id,
];
/**
* Ensure the method does not already exist!!
*/
$token = ClientGatewayToken::where([
'gateway_customer_reference' => $customer->id,
'token' => $method->id,
'client_id' => $this->stripe->client->id,
'company_id' => $this->stripe->client->company_id,
])->first();
2023-10-26 04:57:44 +02:00
if($token) {
return $token;
2023-10-26 04:57:44 +02:00
}
return $this->stripe->storeGatewayToken($data, ['gateway_customer_reference' => $customer->id]);
} catch (Exception $e) {
return $this->stripe->processInternallyFailedPayment($this->stripe, $e);
}
}
public function livewirePaymentView(array $data): string
{
return 'gateways.stripe.ach.pay_livewire';
}
public function paymentData(array $data): array
{
$data['gateway'] = $this->stripe;
$data['currency'] = $this->stripe->client->getCurrencyCode();
$data['payment_method_id'] = GatewayType::BANK_TRANSFER;
$data['customer'] = $this->stripe->findOrCreateCustomer();
$data['amount'] = $this->stripe->convertToStripeAmount($data['total']['amount_with_fee'], $this->stripe->client->currency()->precision, $this->stripe->client->currency());
2024-10-08 01:11:43 +02:00
$data['authorized'] = true;
$description = $this->stripe->getDescription(false);
$intent = false;
if (count($data['tokens']) == 1) {
$token = $data['tokens'][0];
$meta = $token->meta;
if(isset($meta->state) && $meta->state == 'unauthorized') {
2024-10-08 01:11:43 +02:00
$data['authorized'] = false;
// return redirect()->route('client.payment_methods.show', $token->hashed_id);
}
}
if (count($data['tokens']) == 0) {
$intent =
$this->stripe->createPaymentIntent(
[
'amount' => $data['amount'],
'currency' => $data['currency'],
'setup_future_usage' => 'off_session',
'customer' => $data['customer']->id,
'payment_method_types' => ['us_bank_account'],
'description' => $description,
'metadata' => [
'payment_hash' => $this->stripe->payment_hash->hash,
'gateway_type_id' => GatewayType::BANK_TRANSFER,
],
'statement_descriptor' => $this->stripe->getStatementDescriptor(),
]
);
}
$data['client_secret'] = $intent ? $intent->client_secret : false;
return $data;
}
2020-06-09 14:40:55 +02:00
}