mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2024-11-15 07:33:04 +01:00
301 lines
11 KiB
PHP
301 lines
11 KiB
PHP
<?php
|
|
|
|
/**
|
|
* Invoice Ninja (https://invoiceninja.com).
|
|
*
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
*
|
|
* @copyright Copyright (c) 2024. Invoice Ninja LLC (https://invoiceninja.com)
|
|
*
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
|
*/
|
|
|
|
namespace App\PaymentDrivers\Braintree;
|
|
|
|
use App\Exceptions\PaymentFailed;
|
|
use App\Http\Requests\ClientPortal\Payments\PaymentResponseRequest;
|
|
use App\Http\Requests\Request;
|
|
use App\Jobs\Util\SystemLogger;
|
|
use App\Models\GatewayType;
|
|
use App\Models\Payment;
|
|
use App\Models\PaymentType;
|
|
use App\Models\SystemLog;
|
|
use App\PaymentDrivers\BraintreePaymentDriver;
|
|
use App\PaymentDrivers\Common\LivewireMethodInterface;
|
|
|
|
class CreditCard implements LivewireMethodInterface
|
|
{
|
|
/**
|
|
* @var BraintreePaymentDriver
|
|
*/
|
|
private $braintree;
|
|
|
|
public function __construct(BraintreePaymentDriver $braintree)
|
|
{
|
|
$this->braintree = $braintree;
|
|
|
|
$this->braintree->init();
|
|
}
|
|
|
|
public function authorizeView(array $data)
|
|
{
|
|
$data['gateway'] = $this->braintree;
|
|
$data['threeds_enable'] = $this->braintree->company_gateway->getConfigField('threeds') ? "true" : "false";
|
|
|
|
return render('gateways.braintree.credit_card.authorize', $data);
|
|
}
|
|
|
|
public function authorizeResponse($data): \Illuminate\Http\RedirectResponse
|
|
{
|
|
return back();
|
|
}
|
|
|
|
/**
|
|
* Credit card payment page.
|
|
*
|
|
* @param array $data
|
|
* @return array
|
|
*/
|
|
|
|
private function threeDParameters(array $data)
|
|
{
|
|
return [
|
|
'amount' => $data['amount_with_fee'],
|
|
'email' => $this->braintree->client->present()->email(),
|
|
'billingAddress' => [
|
|
'givenName' => $this->braintree->client->present()->first_name() ?: $this->braintree->client->present()->name(),
|
|
'surname' => $this->braintree->client->present()->last_name() ?: '',
|
|
'phoneNumber' => $this->braintree->client->present()->phone(),
|
|
'streetAddress' => $this->braintree->client->address1 ?: '',
|
|
'extendedAddress' => $this->braintree->client->address2 ?: '',
|
|
'locality' => $this->braintree->client->city ?: '',
|
|
'postalCode' => $this->braintree->client->postal_code ?: '',
|
|
'countryCodeAlpha2' => $this->braintree->client->country ? $this->braintree->client->country->iso_3166_2 : 'US',
|
|
]
|
|
];
|
|
}
|
|
|
|
public function paymentView(array $data)
|
|
{
|
|
$data = $this->paymentData($data);
|
|
|
|
return render('gateways.braintree.credit_card.pay', $data);
|
|
}
|
|
|
|
/**
|
|
* Process the credit card payments.
|
|
*
|
|
* @param PaymentResponseRequest $request
|
|
* @return \Illuminate\Http\RedirectResponse|void
|
|
* @throws PaymentFailed
|
|
*/
|
|
public function paymentResponse(PaymentResponseRequest $request)
|
|
{
|
|
$this->braintree->client->fresh();
|
|
|
|
$state = [
|
|
'server_response' => json_decode($request->gateway_response),
|
|
'payment_hash' => $request->payment_hash,
|
|
];
|
|
|
|
$state = array_merge($state, $request->all());
|
|
$state['store_card'] = boolval($state['store_card']);
|
|
|
|
$this->braintree->payment_hash->data = array_merge((array) $this->braintree->payment_hash->data, $state);
|
|
$this->braintree->payment_hash->save();
|
|
|
|
$customer = $this->braintree->findOrCreateCustomer();
|
|
|
|
$token = $this->getPaymentToken($request->all(), $customer->id);
|
|
|
|
$data = [
|
|
'amount' => $this->braintree->payment_hash->data->amount_with_fee, //@phpstan-ignore-line
|
|
'paymentMethodToken' => $token,
|
|
'deviceData' => $state['client-data'],
|
|
'options' => [
|
|
'submitForSettlement' => true,
|
|
],
|
|
'billing' => [
|
|
'streetAddress' => $this->braintree->client->address1 ?: '',
|
|
'extendedAddress' => $this->braintree->client->address2 ?: '',
|
|
'locality' => $this->braintree->client->city ?: '',
|
|
'postalCode' => $this->braintree->client->postal_code ?: '',
|
|
'countryCodeAlpha2' => $this->braintree->client->country ? $this->braintree->client->country->iso_3166_2 : 'US',
|
|
]
|
|
];
|
|
|
|
if ($this->braintree->company_gateway->getConfigField('merchantAccountId')) {
|
|
/** https://developer.paypal.com/braintree/docs/reference/request/transaction/sale/php#full-example */
|
|
$data['merchantAccountId'] = $this->braintree->company_gateway->getConfigField('merchantAccountId');
|
|
}
|
|
|
|
try {
|
|
$result = $this->braintree->gateway->transaction()->sale($data);
|
|
} catch (\Exception $e) {
|
|
if ($e instanceof \Braintree\Exception\Authorization) {
|
|
$this->braintree->sendFailureMail(ctrans('texts.generic_gateway_error'));
|
|
|
|
throw new PaymentFailed(ctrans('texts.generic_gateway_error'), $e->getCode());
|
|
}
|
|
|
|
$this->braintree->sendFailureMail($e->getMessage());
|
|
|
|
throw new PaymentFailed($e->getMessage(), $e->getCode());
|
|
}
|
|
|
|
if ($result->success) {
|
|
$this->braintree->logSuccessfulGatewayResponse(['response' => $request->server_response, 'data' => $this->braintree->payment_hash], SystemLog::TYPE_BRAINTREE);
|
|
|
|
if ($request->store_card && is_null($request->token)) {
|
|
$payment_method = $this->braintree->gateway->paymentMethod()->find($token);
|
|
|
|
$this->storePaymentMethod($payment_method, $customer->id);
|
|
}
|
|
|
|
return $this->processSuccessfulPayment($result);
|
|
}
|
|
|
|
$error = $result ?: 'Undefined gateway error';
|
|
|
|
return $this->processUnsuccessfulPayment($error);
|
|
}
|
|
|
|
private function getPaymentToken(array $data, $customerId): ?string
|
|
{
|
|
if (array_key_exists('token', $data) && ! is_null($data['token'])) {
|
|
return $data['token'];
|
|
}
|
|
|
|
$gateway_response = \json_decode($data['gateway_response']);
|
|
|
|
$data = [
|
|
'customerId' => $customerId,
|
|
'paymentMethodNonce' => $gateway_response->nonce,
|
|
'options' => [
|
|
'verifyCard' => true,
|
|
],
|
|
'billingAddress' => [
|
|
'streetAddress' => $this->braintree->client->address1 ?: '',
|
|
'extendedAddress' => $this->braintree->client->address2 ?: '',
|
|
'locality' => $this->braintree->client->city ?: '',
|
|
'postalCode' => $this->braintree->client->postal_code ?: '',
|
|
'countryCodeAlpha2' => $this->braintree->client->country ? $this->braintree->client->country->iso_3166_2 : 'US',
|
|
]
|
|
];
|
|
|
|
if ($this->braintree->company_gateway->getConfigField('merchantAccountId')) {
|
|
/** https://developer.paypal.com/braintree/docs/reference/request/transaction/sale/php#full-example */
|
|
$data['options']['verificationMerchantAccountId'] = $this->braintree->company_gateway->getConfigField('merchantAccountId');
|
|
}
|
|
|
|
$response = $this->braintree->gateway->paymentMethod()->create($data);
|
|
|
|
if ($response->success) {
|
|
return $response->paymentMethod->token;
|
|
}
|
|
|
|
$this->braintree->sendFailureMail($response->message);
|
|
|
|
throw new PaymentFailed($response->message);
|
|
}
|
|
|
|
private function processSuccessfulPayment($response)
|
|
{
|
|
$state = $this->braintree->payment_hash->data;
|
|
|
|
$data = [
|
|
'payment_type' => PaymentType::parseCardType(strtolower($response->transaction->creditCard['cardType'])),
|
|
'amount' => $this->braintree->payment_hash->data->amount_with_fee,
|
|
'transaction_reference' => $response->transaction->id,
|
|
'gateway_type_id' => GatewayType::CREDIT_CARD,
|
|
];
|
|
|
|
$payment = $this->braintree->createPayment($data, Payment::STATUS_COMPLETED);
|
|
|
|
SystemLogger::dispatch(
|
|
['response' => $response, 'data' => $data],
|
|
SystemLog::CATEGORY_GATEWAY_RESPONSE,
|
|
SystemLog::EVENT_GATEWAY_SUCCESS,
|
|
SystemLog::TYPE_BRAINTREE,
|
|
$this->braintree->client,
|
|
$this->braintree->client->company,
|
|
);
|
|
|
|
return redirect()->route('client.payments.show', ['payment' => $this->braintree->encodePrimaryKey($payment->id)]);
|
|
}
|
|
|
|
/**
|
|
* @throws PaymentFailed
|
|
*/
|
|
private function processUnsuccessfulPayment($response)
|
|
{
|
|
$this->braintree->sendFailureMail($response?->transaction?->additionalProcessorResponse);
|
|
|
|
$message = [
|
|
'server_response' => $response,
|
|
'data' => $this->braintree->payment_hash->data,
|
|
];
|
|
|
|
SystemLogger::dispatch(
|
|
$message,
|
|
SystemLog::CATEGORY_GATEWAY_RESPONSE,
|
|
SystemLog::EVENT_GATEWAY_FAILURE,
|
|
SystemLog::TYPE_BRAINTREE,
|
|
$this->braintree->client,
|
|
$this->braintree->client->company,
|
|
);
|
|
|
|
throw new PaymentFailed($response?->transaction?->additionalProcessorResponse ?: 'Unhandled error, please contact merchant', $response?->transaction?->processorResponseCode ?: 500);
|
|
}
|
|
|
|
private function storePaymentMethod($method, $customer_reference)
|
|
{
|
|
try {
|
|
$payment_meta = new \stdClass();
|
|
$payment_meta->exp_month = (string) $method->expirationMonth;
|
|
$payment_meta->exp_year = (string) $method->expirationYear;
|
|
$payment_meta->brand = (string) $method->cardType;
|
|
$payment_meta->last4 = (string) $method->last4;
|
|
$payment_meta->type = GatewayType::CREDIT_CARD;
|
|
|
|
$data = [
|
|
'payment_meta' => $payment_meta,
|
|
'token' => $method->token,
|
|
'payment_method_id' => $this->braintree->payment_hash->data->payment_method_id,
|
|
];
|
|
|
|
$this->braintree->storeGatewayToken($data, ['gateway_customer_reference' => $customer_reference]);
|
|
} catch (\Exception $e) {
|
|
return $this->braintree->processInternallyFailedPayment($this->braintree, $e);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
public function livewirePaymentView(array $data): string
|
|
{
|
|
return 'gateways.braintree.credit_card.pay_livewire';
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
public function paymentData(array $data): array
|
|
{
|
|
$data['gateway'] = $this->braintree;
|
|
$data['client_token'] = $this->braintree->gateway->clientToken()->generate();
|
|
$data['threeds'] = $this->threeDParameters($data);
|
|
$data['threeds_enable'] = $this->braintree->company_gateway->getConfigField('threeds') ? "true" : "false";
|
|
|
|
if ($this->braintree->company_gateway->getConfigField('merchantAccountId')) {
|
|
/** https://developer.paypal.com/braintree/docs/reference/request/client-token/generate#merchant_account_id */
|
|
$data['client_token'] = $this->braintree->gateway->clientToken()->generate([ // @phpstan-ignore-line
|
|
'merchantAccountId' => $this->braintree->company_gateway->getConfigField('merchantAccountId'),
|
|
]);
|
|
}
|
|
|
|
return $data;
|
|
}
|
|
}
|