mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2024-11-15 07:33:04 +01:00
527 lines
17 KiB
PHP
527 lines
17 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;
|
|
|
|
use Exception;
|
|
use App\Models\Client;
|
|
use Braintree\Gateway;
|
|
use App\Models\Invoice;
|
|
use App\Models\Payment;
|
|
use App\Models\SystemLog;
|
|
use App\Models\GatewayType;
|
|
use App\Models\PaymentHash;
|
|
use App\Models\PaymentType;
|
|
use App\Models\ClientContact;
|
|
use App\Factory\ClientFactory;
|
|
use Illuminate\Support\Carbon;
|
|
use App\Jobs\Util\SystemLogger;
|
|
use App\Models\ClientGatewayToken;
|
|
use App\Factory\ClientContactFactory;
|
|
use App\PaymentDrivers\Braintree\ACH;
|
|
use App\Utils\Traits\GeneratesCounter;
|
|
use Illuminate\Database\QueryException;
|
|
use App\PaymentDrivers\Braintree\PayPal;
|
|
use Illuminate\Support\Facades\Validator;
|
|
use App\PaymentDrivers\Braintree\CreditCard;
|
|
|
|
class BraintreePaymentDriver extends BaseDriver
|
|
{
|
|
use GeneratesCounter;
|
|
|
|
public $refundable = true;
|
|
|
|
public $token_billing = true;
|
|
|
|
public $can_authorise_credit_card = true;
|
|
|
|
private bool $completed = true;
|
|
|
|
/**
|
|
* @var Gateway;
|
|
*/
|
|
public Gateway $gateway;
|
|
|
|
public static $methods = [
|
|
GatewayType::CREDIT_CARD => CreditCard::class,
|
|
GatewayType::PAYPAL => PayPal::class,
|
|
GatewayType::BANK_TRANSFER => ACH::class,
|
|
];
|
|
|
|
public const SYSTEM_LOG_TYPE = SystemLog::TYPE_BRAINTREE;
|
|
|
|
public function init(): self
|
|
{
|
|
$this->gateway = new Gateway([
|
|
'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'),
|
|
]);
|
|
|
|
return $this;
|
|
}
|
|
|
|
public function setPaymentMethod($payment_method_id)
|
|
{
|
|
$class = self::$methods[$payment_method_id];
|
|
|
|
$this->payment_method = new $class($this);
|
|
|
|
return $this;
|
|
}
|
|
|
|
public function gatewayTypes(): array
|
|
{
|
|
$types = [
|
|
GatewayType::PAYPAL,
|
|
GatewayType::CREDIT_CARD,
|
|
GatewayType::BANK_TRANSFER,
|
|
];
|
|
|
|
return $types;
|
|
}
|
|
|
|
public function authorizeView($data)
|
|
{
|
|
return $this->payment_method->authorizeView($data);
|
|
}
|
|
|
|
public function authorizeResponse($data)
|
|
{
|
|
return $this->payment_method->authorizeResponse($data);
|
|
}
|
|
|
|
public function processPaymentView(array $data)
|
|
{
|
|
return $this->payment_method->paymentView($data);
|
|
}
|
|
|
|
public function processPaymentResponse($request)
|
|
{
|
|
return $this->payment_method->paymentResponse($request);
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
$customer = $this->searchByEmail();
|
|
|
|
if ($customer) {
|
|
return $customer;
|
|
}
|
|
|
|
$result = $this->gateway->customer()->create([
|
|
'firstName' => $this->client->present()->name(),
|
|
'email' => $this->client->present()->email(),
|
|
'phone' => $this->client->present()->phone(),
|
|
]);
|
|
|
|
if ($result->success) {
|
|
$address = $this->gateway->address()->create([
|
|
'customerId' => $result->customer->id,
|
|
'firstName' => $this->client->present()->name(),
|
|
'streetAddress' => $this->client->address1 ?: '',
|
|
'postalCode' => $this->client->postal_code ?: '',
|
|
'countryCodeAlpha2' => $this->client->country ? $this->client->country->iso_3166_2 : '',
|
|
]);
|
|
|
|
return $result->customer;
|
|
}
|
|
//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,
|
|
];
|
|
|
|
SystemLogger::dispatch(['server_response' => $result, 'data' => $data], SystemLog::CATEGORY_GATEWAY_RESPONSE, SystemLog::EVENT_GATEWAY_FAILURE, SystemLog::TYPE_BRAINTREE, $this->client, $this->client->company);
|
|
}
|
|
|
|
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 refund(Payment $payment, $amount, $return_client_response = false)
|
|
{
|
|
$this->init();
|
|
|
|
try {
|
|
$response = $this->gateway->transaction()->refund($payment->transaction_reference, $amount);
|
|
} catch (Exception $e) {
|
|
$data = [
|
|
'transaction_reference' => null,
|
|
'transaction_response' => json_encode($e->getMessage()),
|
|
'success' => false,
|
|
'description' => $e->getMessage(),
|
|
'code' => $e->getCode(),
|
|
];
|
|
|
|
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;
|
|
}
|
|
|
|
if ($response->success) {
|
|
$data = [
|
|
'transaction_reference' => $payment->transaction_reference,
|
|
'transaction_response' => json_encode($response),
|
|
'success' => (bool) $response->success,
|
|
'description' => ctrans('texts.plan_refunded'),
|
|
'code' => 0,
|
|
];
|
|
|
|
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;
|
|
} else {
|
|
$error = $response->errors->deepAll()[0];
|
|
|
|
$data = [
|
|
'transaction_reference' => null,
|
|
'transaction_response' => $response->errors->deepAll(),
|
|
'success' => false,
|
|
'description' => $error->message,
|
|
'code' => $error->code,
|
|
];
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
public function tokenBilling(ClientGatewayToken $cgt, PaymentHash $payment_hash)
|
|
{
|
|
$amount = array_sum(array_column($payment_hash->invoices(), 'amount')) + $payment_hash->fee_total;
|
|
|
|
$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,
|
|
$this->client,
|
|
$this->client->company,
|
|
);
|
|
|
|
return $payment;
|
|
}
|
|
|
|
if (! $result->success) {
|
|
$this->unWindGatewayFees($payment_hash);
|
|
|
|
$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,
|
|
$this->client,
|
|
$this->client->company
|
|
);
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Required fields for client to fill, to proceed with gateway actions.
|
|
*
|
|
* @return array[]
|
|
*/
|
|
public function getClientRequiredFields(): array
|
|
{
|
|
$fields = [];
|
|
|
|
$fields[] = ['name' => 'contact_first_name', 'label' => ctrans('texts.first_name'), 'type' => 'text', 'validation' => 'required'];
|
|
$fields[] = ['name' => 'contact_last_name', 'label' => ctrans('texts.last_name'), 'type' => 'text', 'validation' => 'required'];
|
|
$fields[] = ['name' => 'contact_email', 'label' => ctrans('texts.email'), 'type' => 'text', 'validation' => 'required,email:rfc'];
|
|
$fields[] = ['name' => 'client_address_line_1', 'label' => ctrans('texts.address1'), 'type' => 'text', 'validation' => 'required'];
|
|
$fields[] = ['name' => 'client_city', 'label' => ctrans('texts.city'), 'type' => 'text', 'validation' => 'required'];
|
|
$fields[] = ['name' => 'client_state', 'label' => ctrans('texts.state'), 'type' => 'text', 'validation' => 'required'];
|
|
$fields[] = ['name' => 'client_country_id', 'label' => ctrans('texts.country'), 'type' => 'text', 'validation' => 'required'];
|
|
|
|
return $fields;
|
|
}
|
|
|
|
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(
|
|
$request->input('bt_signature'),
|
|
$request->input('bt_payload')
|
|
);
|
|
|
|
nlog('braintree webhook');
|
|
|
|
return response()->json([], 200);
|
|
}
|
|
|
|
public function auth(): bool
|
|
{
|
|
|
|
try {
|
|
$ct = $this->init()->gateway->clientToken()->generate();
|
|
|
|
return true;
|
|
} catch(\Exception $e) {
|
|
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private function find(string $customer_id = '')
|
|
{
|
|
|
|
try {
|
|
return $this->init()->gateway->customer()->find($customer_id);
|
|
} catch(\Exception $e) {
|
|
return false;
|
|
}
|
|
|
|
}
|
|
|
|
private function findTokens(string $gateway_customer_reference)
|
|
{
|
|
return ClientGatewayToken::where('company_id', $this->company_gateway->company_id)
|
|
->where('gateway_customer_reference', $gateway_customer_reference)
|
|
->exists();
|
|
}
|
|
|
|
private function getToken(string $token, string $gateway_customer_reference)
|
|
{
|
|
|
|
return ClientGatewayToken::where('company_id', $this->company_gateway->company_id)
|
|
->where('gateway_customer_reference', $gateway_customer_reference)
|
|
->where('token', $token)
|
|
->first();
|
|
|
|
}
|
|
|
|
private function findClient(string $email)
|
|
{
|
|
return ClientContact::where('company_id', $this->company_gateway->company_id)
|
|
->where('email', $email)
|
|
->first()->client ?? false;
|
|
}
|
|
|
|
private function addClientCards(Client $client, array $cards)
|
|
{
|
|
|
|
$this->client = $client;
|
|
|
|
foreach($cards as $card) {
|
|
|
|
if($this->getToken($card->token, $card->customerId) || Carbon::createFromDate($card->expirationYear, $card->expirationMonth, '1')->lt(now())) {
|
|
continue;
|
|
}
|
|
|
|
$payment_meta = new \stdClass();
|
|
$payment_meta->exp_month = (string) $card->expirationMonth;
|
|
$payment_meta->exp_year = (string) $card->expirationYear;
|
|
$payment_meta->brand = (string) $card->cardType;
|
|
$payment_meta->last4 = (string) $card->last4;
|
|
$payment_meta->type = GatewayType::CREDIT_CARD;
|
|
|
|
$data = [
|
|
'payment_meta' => $payment_meta,
|
|
'token' => $card->token,
|
|
'payment_method_id' => GatewayType::CREDIT_CARD,
|
|
];
|
|
|
|
$this->storeGatewayToken($data, ['gateway_customer_reference' => $card->customerId]);
|
|
|
|
nlog("adding card to customer payment profile");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
public function createNinjaClient(mixed $customer): Client
|
|
{
|
|
|
|
$client = ClientFactory::create($this->company_gateway->company_id, $this->company_gateway->user_id);
|
|
|
|
$b_business_address = count($customer->addresses) >= 1 ? $customer->addresses[0] : false;
|
|
$b_shipping_address = count($customer->addresses) > 1 ? $customer->addresses[1] : false;
|
|
$import_client_data = [];
|
|
|
|
if($b_business_address) {
|
|
|
|
$braintree_address =
|
|
[
|
|
'address1' => $b_business_address->extendedAddress ?? '',
|
|
'address2' => $b_business_address->streetAddress ?? '',
|
|
'city' => $b_business_address->locality ?? '',
|
|
'postal_code' => $b_business_address->postalCode ?? '',
|
|
'state' => $b_business_address->region ?? '',
|
|
'country_id' => $b_business_address->countryCodeNumeric ?? '840',
|
|
];
|
|
|
|
$import_client_data = array_merge($import_client_data, $braintree_address);
|
|
}
|
|
|
|
if($b_shipping_address) {
|
|
|
|
$braintree_shipping_address =
|
|
[
|
|
'shipping_address1' => $b_shipping_address->extendedAddress ?? '',
|
|
'shipping_address2' => $b_shipping_address->streetAddress ?? '',
|
|
'shipping_city' => $b_shipping_address->locality ?? '',
|
|
'shipping_postal_code' => $b_shipping_address->postalCode ?? '',
|
|
'shipping_state' => $b_shipping_address->region ?? '',
|
|
'shipping_country_id' => $b_shipping_address->countryCodeNumeric ?? '840',
|
|
];
|
|
|
|
$import_client_data = array_merge($import_client_data, $braintree_shipping_address);
|
|
|
|
}
|
|
|
|
$client->fill($import_client_data);
|
|
|
|
$client->phone = $customer->phone ?? '';
|
|
$client->name = $customer->company ?? $customer->firstName;
|
|
|
|
$settings = $client->settings;
|
|
$settings->currency_id = (string) $this->company_gateway->company->settings->currency_id;
|
|
$client->settings = $settings;
|
|
$client->save();
|
|
|
|
$contact = ClientContactFactory::create($this->company_gateway->company_id, $this->company_gateway->user_id);
|
|
$contact->first_name = $customer->firstName ?? '';
|
|
$contact->last_name = $customer->lastName ?? '';
|
|
$contact->email = $customer->email ?? '';
|
|
$contact->phone = $customer->phone ?? '';
|
|
$contact->client_id = $client->id;
|
|
$contact->saveQuietly();
|
|
|
|
if (! isset($client->number) || empty($client->number)) {
|
|
$x = 1;
|
|
|
|
do {
|
|
try {
|
|
$client->number = $this->getNextClientNumber($client);
|
|
$client->saveQuietly();
|
|
|
|
$this->completed = false;
|
|
} catch (QueryException $e) {
|
|
$x++;
|
|
|
|
if ($x > 10) {
|
|
$this->completed = false;
|
|
}
|
|
}
|
|
} while ($this->completed);
|
|
} else {
|
|
$client->saveQuietly();
|
|
}
|
|
|
|
return $client;
|
|
|
|
}
|
|
|
|
public function importCustomers()
|
|
{
|
|
$customers = $this->init()->gateway->customer()->all();
|
|
|
|
foreach($customers as $c) {
|
|
|
|
$customer = $this->find($c->id);
|
|
|
|
// nlog(count($customer->creditCards). " Exists for {$c->id}");
|
|
|
|
if(!$customer) {
|
|
continue;
|
|
}
|
|
|
|
$client = $this->findClient($customer->email);
|
|
|
|
if(!$this->findTokens($c->id) && !$client) {
|
|
//customer is not referenced in the system - create client
|
|
$client = $this->createNinjaClient($customer);
|
|
// nlog("Creating new Client");
|
|
}
|
|
|
|
$this->addClientCards($client, $customer->creditCards);
|
|
|
|
// nlog("Adding Braintree Client: {$c->id} => {$client->id}");
|
|
|
|
}
|
|
}
|
|
}
|