1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-11-14 07:02:34 +01:00
invoiceninja/app/PaymentDrivers/BraintreePaymentDriver.php

337 lines
11 KiB
PHP
Raw Normal View History

2021-04-28 12:54:27 +02:00
<?php
/**
* Invoice Ninja (https://invoiceninja.com).
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
2023-01-28 23:21:40 +01:00
* @copyright Copyright (c) 2023. Invoice Ninja LLC (https://invoiceninja.com)
2021-04-28 12:54:27 +02:00
*
2021-06-16 08:58:16 +02:00
* @license https://www.elastic.co/licensing/elastic-license
2021-04-28 12:54:27 +02:00
*/
namespace App\PaymentDrivers;
use App\Jobs\Util\SystemLogger;
2021-04-29 15:05:45 +02:00
use App\Models\ClientGatewayToken;
2021-04-28 12:54:27 +02:00
use App\Models\GatewayType;
use App\Models\Invoice;
2021-05-05 17:24:31 +02:00
use App\Models\Payment;
use App\Models\PaymentHash;
use App\Models\PaymentType;
2021-04-28 12:54:27 +02:00
use App\Models\SystemLog;
2021-08-26 15:38:28 +02:00
use App\PaymentDrivers\Braintree\ACH;
2021-04-28 12:54:27 +02:00
use App\PaymentDrivers\Braintree\CreditCard;
2021-05-03 14:10:46 +02:00
use App\PaymentDrivers\Braintree\PayPal;
use Braintree\Gateway;
use Exception;
2022-01-09 06:30:39 +01:00
use Illuminate\Support\Facades\Validator;
2021-04-28 12:54:27 +02:00
class BraintreePaymentDriver extends BaseDriver
{
public $refundable = true;
public $token_billing = true;
public $can_authorise_credit_card = true;
2021-04-28 14:54:50 +02:00
/**
* @var Gateway;
2021-04-28 14:54:50 +02:00
*/
2021-10-25 17:44:55 +02:00
public Gateway $gateway;
2021-04-28 12:54:27 +02:00
public static $methods = [
GatewayType::CREDIT_CARD => CreditCard::class,
2021-05-04 14:56:30 +02:00
GatewayType::PAYPAL => PayPal::class,
2021-08-26 15:38:28 +02:00
GatewayType::BANK_TRANSFER => ACH::class,
2021-04-28 12:54:27 +02:00
];
const SYSTEM_LOG_TYPE = SystemLog::TYPE_BRAINTREE;
2023-03-14 07:56:04 +01:00
public function init(): self
2021-04-28 12:54:27 +02:00
{
$this->gateway = new Gateway([
2021-04-28 14:54:50 +02:00
'environment' => $this->company_gateway->getConfigField('testMode') ? 'sandbox' : 'production',
'merchantId' => $this->company_gateway->getConfigField('merchantId'),
'publicKey' => $this->company_gateway->getConfigField('publicKey'),
'privateKey' => $this->company_gateway->getConfigField('privateKey'),
]);
2023-03-14 07:56:04 +01:00
return $this;
2021-04-28 12:54:27 +02:00
}
public function setPaymentMethod($payment_method_id)
{
$class = self::$methods[$payment_method_id];
2021-05-03 15:48:53 +02:00
$this->payment_method = new $class($this);
2021-04-28 12:54:27 +02:00
return $this;
}
public function gatewayTypes(): array
{
$types = [
2021-04-28 12:54:27 +02:00
GatewayType::PAYPAL,
2021-08-26 15:38:28 +02:00
GatewayType::CREDIT_CARD,
GatewayType::BANK_TRANSFER,
2021-04-28 12:54:27 +02:00
];
2021-08-26 15:38:28 +02:00
return $types;
2021-04-28 12:54:27 +02:00
}
2021-04-28 15:03:22 +02:00
public function authorizeView($data)
{
return $this->payment_method->authorizeView($data);
}
public function authorizeResponse($data)
{
return $this->payment_method->authorizeResponse($data);
}
2021-04-28 12:54:27 +02:00
public function processPaymentView(array $data)
{
return $this->payment_method->paymentView($data);
}
2021-04-28 14:54:50 +02:00
public function processPaymentResponse($request)
{
return $this->payment_method->paymentResponse($request);
}
2021-04-29 15:05:45 +02:00
public function findOrCreateCustomer()
{
$existing = ClientGatewayToken::query()
->where('company_gateway_id', $this->company_gateway->id)
->where('client_id', $this->client->id)
->first();
if ($existing) {
return $this->gateway->customer()->find($existing->gateway_customer_reference);
}
2023-03-14 07:56:04 +01:00
$customer = $this->searchByEmail();
if ($customer) {
return $customer;
}
$result = $this->gateway->customer()->create([
2022-09-05 05:01:13 +02:00
'firstName' => $this->client->present()->name(),
'email' => $this->client->present()->email(),
'phone' => $this->client->present()->phone(),
2021-04-29 15:05:45 +02:00
]);
if ($result->success) {
2021-10-25 17:45:25 +02:00
$address = $this->gateway->address()->create([
'customerId' => $result->customer->id,
2022-09-05 05:01:13 +02:00
'firstName' => $this->client->present()->name(),
'streetAddress' => $this->client->address1 ?: '',
'postalCode' => $this->client->postal_code ?: '',
'countryCodeAlpha2' => $this->client->country ? $this->client->country->iso_3166_2 : '',
]);
2021-04-29 15:05:45 +02:00
return $result->customer;
}
2023-02-16 02:36:09 +01:00
//12-08-2022 catch when the customer is not created.
$data = [
'transaction_reference' => null,
'transaction_response' => $result,
'success' => false,
'description' => 'Could not create customer',
'code' => 500,
];
2023-02-16 02:36:09 +01:00
SystemLogger::dispatch(['server_response' => $result, 'data' => $data], SystemLog::CATEGORY_GATEWAY_RESPONSE, SystemLog::EVENT_GATEWAY_FAILURE, SystemLog::TYPE_BRAINTREE, $this->client, $this->client->company);
2021-04-29 15:05:45 +02:00
}
2023-03-14 07:56:04 +01:00
private function searchByEmail()
{
$result = $this->gateway->customer()->search([
\Braintree\CustomerSearch::email()->is($this->client->present()->email()),
]);
if ($result->maximumCount() > 0) {
return $result->firstItem();
}
}
// public function updateCustomer()
// {
// $customer = $this->findOrCreateCustomer();
// $result = $this->gateway->customer()->update(
// $customer->id,
// [
// 'firstName' => $this->client->present()->name(),
// 'email' => $this->client->present()->email(),
// 'phone' => $this->client->present()->phone(),
// 'creditCard' => [
// 'billingAddress' => [
// 'options' => [
// 'updateExisting' => true
// ],
// 'firstName' => $this->client->present()->first_name() ?: $this->client->present()->name(),
// 'lastName' => $this->client->present()->last_name() ?: '',
// 'streetAddress' => $this->client->address1 ?: '',
// 'extendedAddress' =>$this->client->address2 ?: '',
// 'locality' => $this->client->city ?: '',
// 'postalCode' => $this->client->postal_code ?: '',
// 'countryCodeAlpha2' => $this->client->country ? $this->client->country->iso_3166_2 : 'US',
// ],
// ],
// ]
// );
// }
2021-05-05 17:24:31 +02:00
public function refund(Payment $payment, $amount, $return_client_response = false)
{
$this->init();
2021-10-25 17:45:25 +02:00
try {
2021-05-05 17:24:31 +02:00
$response = $this->gateway->transaction()->refund($payment->transaction_reference, $amount);
2021-08-25 06:36:30 +02:00
} catch (Exception $e) {
$data = [
'transaction_reference' => null,
'transaction_response' => json_encode($e->getMessage()),
'success' => false,
'description' => $e->getMessage(),
'code' => $e->getCode(),
];
2021-08-26 15:38:28 +02:00
2021-08-25 06:36:30 +02:00
SystemLogger::dispatch(['server_response' => null, 'data' => $data], SystemLog::CATEGORY_GATEWAY_RESPONSE, SystemLog::EVENT_GATEWAY_FAILURE, SystemLog::TYPE_BRAINTREE, $this->client, $this->client->company);
return $data;
}
2021-08-26 15:38:28 +02:00
2021-10-25 17:45:25 +02:00
if ($response->success) {
2021-08-25 06:36:30 +02:00
$data = [
2021-10-25 17:44:34 +02:00
'transaction_reference' => $payment->transaction_reference,
2021-05-06 16:38:54 +02:00
'transaction_response' => json_encode($response),
2021-10-25 17:44:34 +02:00
'success' => (bool) $response->success,
'description' => ctrans('texts.plan_refunded'),
2021-05-06 16:38:54 +02:00
'code' => 0,
];
2021-08-25 06:36:30 +02:00
SystemLogger::dispatch(['server_response' => $response, 'data' => $data], SystemLog::CATEGORY_GATEWAY_RESPONSE, SystemLog::EVENT_GATEWAY_SUCCESS, SystemLog::TYPE_BRAINTREE, $this->client, $this->client->company);
return $data;
2021-10-25 17:45:25 +02:00
} else {
2021-08-25 06:36:30 +02:00
$error = $response->errors->deepAll()[0];
$data = [
2021-05-06 16:38:54 +02:00
'transaction_reference' => null,
2021-08-25 06:36:30 +02:00
'transaction_response' => $response->errors->deepAll(),
2021-05-06 16:38:54 +02:00
'success' => false,
2021-08-25 06:36:30 +02:00
'description' => $error->message,
'code' => $error->code,
2021-05-06 16:38:54 +02:00
];
2021-08-25 06:36:30 +02:00
SystemLogger::dispatch(['server_response' => $response, 'data' => $data], SystemLog::CATEGORY_GATEWAY_RESPONSE, SystemLog::EVENT_GATEWAY_FAILURE, SystemLog::TYPE_BRAINTREE, $this->client, $this->client->company);
return $data;
2021-05-05 17:24:31 +02:00
}
}
public function tokenBilling(ClientGatewayToken $cgt, PaymentHash $payment_hash)
{
$amount = array_sum(array_column($payment_hash->invoices(), 'amount')) + $payment_hash->fee_total;
2023-08-06 09:03:12 +02:00
$invoice = Invoice::query()->whereIn('id', $this->transformKeys(array_column($payment_hash->invoices(), 'invoice_id')))->withTrashed()->first();
if ($invoice) {
$description = "Invoice {$invoice->number} for {$amount} for client {$this->client->present()->name()}";
} else {
$description = "Payment with no invoice for amount {$amount} for client {$this->client->present()->name()}";
}
$this->init();
$result = $this->gateway->transaction()->sale([
'amount' => $amount,
'paymentMethodToken' => $cgt->token,
'deviceData' => '',
'options' => [
'submitForSettlement' => true,
],
]);
if ($result->success) {
$this->confirmGatewayFee();
$data = [
'payment_type' => PaymentType::parseCardType(strtolower($result->transaction->creditCard['cardType'])),
'amount' => $amount,
'transaction_reference' => $result->transaction->id,
'gateway_type_id' => GatewayType::CREDIT_CARD,
];
$payment = $this->createPayment($data, Payment::STATUS_COMPLETED);
SystemLogger::dispatch(
['response' => $result, 'data' => $data],
SystemLog::CATEGORY_GATEWAY_RESPONSE,
SystemLog::EVENT_GATEWAY_SUCCESS,
SystemLog::TYPE_BRAINTREE,
2021-08-30 12:43:00 +02:00
$this->client,
$this->client->company,
);
return $payment;
}
if (! $result->success) {
$this->unWindGatewayFees($payment_hash);
2021-10-17 12:40:40 +02:00
$this->sendFailureMail($result->transaction->additionalProcessorResponse);
$message = [
'server_response' => $result,
'data' => $this->payment_hash->data,
];
SystemLogger::dispatch(
$message,
SystemLog::CATEGORY_GATEWAY_RESPONSE,
SystemLog::EVENT_GATEWAY_FAILURE,
SystemLog::TYPE_BRAINTREE,
2021-08-30 12:43:00 +02:00
$this->client,
$this->client->company
);
return false;
}
}
2022-01-09 06:30:39 +01:00
public function processWebhookRequest($request)
{
$validator = Validator::make($request->all(), [
'bt_signature' => ['required'],
'bt_payload' => ['required'],
]);
if ($validator->fails()) {
return response()->json($validator->errors(), 422);
}
$this->init();
$webhookNotification = $this->gateway->webhookNotification()->parse(
2023-02-16 02:36:09 +01:00
$request->input('bt_signature'),
$request->input('bt_payload')
2022-01-09 06:30:39 +01:00
);
nlog('braintree webhook');
2022-01-09 07:01:41 +01:00
2022-02-17 04:02:23 +01:00
// if($webhookNotification)
// nlog($webhookNotification->kind);
2022-01-09 07:01:41 +01:00
// // Example values for webhook notification properties
// $message = $webhookNotification->kind; // "subscription_went_past_due"
// $message = $webhookNotification->timestamp->format('D M j G:i:s T Y'); // "Sun Jan 1 00:00:00 UTC 2012"
2022-01-09 06:30:39 +01:00
return response()->json([], 200);
}
2021-04-28 12:54:27 +02:00
}