2021-08-14 10:11:45 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Invoice Ninja (https://invoiceninja.com).
|
|
|
|
*
|
|
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
|
|
*
|
|
|
|
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
|
|
|
|
*
|
|
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace App\PaymentDrivers\Square;
|
|
|
|
|
2021-08-19 13:34:45 +02:00
|
|
|
use App\Http\Requests\ClientPortal\Payments\PaymentResponseRequest;
|
|
|
|
use App\Models\ClientGatewayToken;
|
2021-08-14 10:11:45 +02:00
|
|
|
use App\Models\GatewayType;
|
|
|
|
use App\Models\Payment;
|
|
|
|
use App\Models\PaymentType;
|
|
|
|
use App\PaymentDrivers\SquarePaymentDriver;
|
|
|
|
use App\Utils\Traits\MakesHash;
|
|
|
|
use Illuminate\Support\Str;
|
2021-08-18 17:23:31 +02:00
|
|
|
use Square\Http\ApiResponse;
|
2021-08-14 10:11:45 +02:00
|
|
|
|
|
|
|
class CreditCard
|
|
|
|
{
|
|
|
|
use MakesHash;
|
|
|
|
|
2021-08-14 13:00:52 +02:00
|
|
|
public $square_driver;
|
2021-08-14 10:11:45 +02:00
|
|
|
|
2021-08-14 13:00:52 +02:00
|
|
|
public function __construct(SquarePaymentDriver $square_driver)
|
2021-08-14 10:11:45 +02:00
|
|
|
{
|
2021-08-14 13:00:52 +02:00
|
|
|
$this->square_driver = $square_driver;
|
2021-08-14 13:37:04 +02:00
|
|
|
$this->square_driver->init();
|
2021-08-14 10:11:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public function authorizeView($data)
|
|
|
|
{
|
2021-08-14 13:00:52 +02:00
|
|
|
$data['gateway'] = $this->square_driver;
|
2021-08-14 10:11:45 +02:00
|
|
|
|
|
|
|
return render('gateways.square.credit_card.authorize', $data);
|
|
|
|
}
|
|
|
|
|
2021-08-14 13:37:04 +02:00
|
|
|
public function authorizeResponse($request)
|
2021-08-14 10:11:45 +02:00
|
|
|
{
|
2021-08-15 12:39:05 +02:00
|
|
|
/* Step one - process a $1 payment - but don't complete it*/
|
2021-08-14 14:17:35 +02:00
|
|
|
$payment = false;
|
|
|
|
|
2021-08-14 13:00:52 +02:00
|
|
|
$amount_money = new \Square\Models\Money();
|
|
|
|
$amount_money->setAmount(100); //amount in cents
|
|
|
|
$amount_money->setCurrency($this->square_driver->client->currency()->code);
|
|
|
|
|
|
|
|
$body = new \Square\Models\CreatePaymentRequest(
|
|
|
|
$request->sourceId,
|
|
|
|
Str::random(32),
|
|
|
|
$amount_money
|
2021-08-14 10:11:45 +02:00
|
|
|
);
|
|
|
|
|
2021-08-14 13:00:52 +02:00
|
|
|
$body->setAutocomplete(false);
|
|
|
|
$body->setLocationId($this->square_driver->company_gateway->getConfigField('locationId'));
|
|
|
|
$body->setReferenceId(Str::random(16));
|
|
|
|
|
2021-08-14 13:37:04 +02:00
|
|
|
$api_response = $this->square_driver->square->getPaymentsApi()->createPayment($body);
|
2021-08-14 10:11:45 +02:00
|
|
|
|
|
|
|
if ($api_response->isSuccess()) {
|
2021-08-14 14:17:35 +02:00
|
|
|
$result = $api_response->getBody();
|
|
|
|
$payment = json_decode($result);
|
2021-08-14 10:11:45 +02:00
|
|
|
} else {
|
|
|
|
$errors = $api_response->getErrors();
|
2021-08-15 12:39:05 +02:00
|
|
|
return $this->processUnsuccessfulPayment($errors);
|
2021-08-14 10:11:45 +02:00
|
|
|
}
|
|
|
|
|
2021-08-15 13:14:18 +02:00
|
|
|
|
2021-08-15 12:39:05 +02:00
|
|
|
/* Step 3 create the card */
|
2021-08-14 13:37:04 +02:00
|
|
|
$card = new \Square\Models\Card();
|
|
|
|
$card->setCardholderName($this->square_driver->client->present()->name());
|
2021-08-15 13:14:18 +02:00
|
|
|
// $card->setBillingAddress($billing_address);
|
|
|
|
$card->setCustomerId($this->findOrCreateClient());
|
2021-08-14 13:37:04 +02:00
|
|
|
$card->setReferenceId(Str::random(8));
|
2021-08-14 13:00:52 +02:00
|
|
|
|
2021-08-14 13:37:04 +02:00
|
|
|
$body = new \Square\Models\CreateCardRequest(
|
|
|
|
Str::random(32),
|
2021-08-14 14:17:35 +02:00
|
|
|
$payment->payment->id,
|
2021-08-14 13:37:04 +02:00
|
|
|
$card
|
|
|
|
);
|
2021-08-14 13:00:52 +02:00
|
|
|
|
2021-08-14 13:37:04 +02:00
|
|
|
$api_response = $this->square_driver
|
|
|
|
->square
|
|
|
|
->getCardsApi()
|
|
|
|
->createCard($body);
|
2021-08-14 13:00:52 +02:00
|
|
|
|
2021-08-15 12:27:52 +02:00
|
|
|
$card = false;
|
|
|
|
|
2021-08-14 13:37:04 +02:00
|
|
|
if ($api_response->isSuccess()) {
|
2021-08-15 12:27:52 +02:00
|
|
|
$card = $api_response->getBody();
|
|
|
|
$card = json_decode($card);
|
2021-08-14 13:37:04 +02:00
|
|
|
} else {
|
|
|
|
$errors = $api_response->getErrors();
|
2021-08-14 13:00:52 +02:00
|
|
|
|
2021-08-15 12:39:05 +02:00
|
|
|
return $this->processUnsuccessfulPayment($errors);
|
|
|
|
}
|
2021-08-14 13:00:52 +02:00
|
|
|
|
2021-08-15 12:39:05 +02:00
|
|
|
/* Create the token in Invoice Ninja*/
|
2021-08-14 13:00:52 +02:00
|
|
|
$cgt = [];
|
2021-08-15 12:27:52 +02:00
|
|
|
$cgt['token'] = $card->card->id;
|
2021-08-14 13:00:52 +02:00
|
|
|
$cgt['payment_method_id'] = GatewayType::CREDIT_CARD;
|
|
|
|
|
|
|
|
$payment_meta = new \stdClass;
|
2021-08-15 12:27:52 +02:00
|
|
|
$payment_meta->exp_month = $card->card->exp_month;
|
|
|
|
$payment_meta->exp_year = $card->card->exp_year;
|
|
|
|
$payment_meta->brand = $card->card->card_brand;
|
|
|
|
$payment_meta->last4 = $card->card->last_4;
|
2021-08-14 13:00:52 +02:00
|
|
|
$payment_meta->type = GatewayType::CREDIT_CARD;
|
|
|
|
|
|
|
|
$cgt['payment_meta'] = $payment_meta;
|
|
|
|
|
2021-08-19 13:34:45 +02:00
|
|
|
$token = $this->square_driver->storeGatewayToken($cgt, [
|
|
|
|
'gateway_customer_reference' => $this->findOrCreateClient(),
|
|
|
|
]);
|
2021-08-14 13:00:52 +02:00
|
|
|
|
2021-08-15 12:27:52 +02:00
|
|
|
return redirect()->route('client.payment_methods.index');
|
2021-08-14 10:11:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public function paymentView($data)
|
|
|
|
{
|
2021-08-14 13:00:52 +02:00
|
|
|
$data['gateway'] = $this->square_driver;
|
2021-08-14 10:11:45 +02:00
|
|
|
|
2021-08-16 03:11:08 +02:00
|
|
|
|
|
|
|
return render('gateways.square.credit_card.pay', $data);
|
2021-08-14 10:11:45 +02:00
|
|
|
}
|
|
|
|
|
2021-08-19 13:34:45 +02:00
|
|
|
public function paymentResponse(PaymentResponseRequest $request)
|
2021-08-14 10:11:45 +02:00
|
|
|
{
|
2021-08-19 13:34:45 +02:00
|
|
|
$token = $request->sourceId;
|
2021-08-19 13:50:34 +02:00
|
|
|
|
|
|
|
$amount = $this->square_driver->convertAmount(
|
|
|
|
$this->square_driver->payment_hash->data->amount_with_fee
|
|
|
|
);
|
2021-08-19 13:34:45 +02:00
|
|
|
|
|
|
|
if ($request->shouldUseToken()) {
|
|
|
|
$cgt = ClientGatewayToken::where('token', $request->token)->first();
|
|
|
|
$token = $cgt->token;
|
|
|
|
}
|
|
|
|
|
2021-08-18 17:23:31 +02:00
|
|
|
$amount_money = new \Square\Models\Money();
|
2021-08-19 13:50:34 +02:00
|
|
|
$amount_money->setAmount($amount);
|
2021-08-18 17:23:31 +02:00
|
|
|
$amount_money->setCurrency($this->square_driver->client->currency()->code);
|
|
|
|
|
2021-08-19 13:34:45 +02:00
|
|
|
$body = new \Square\Models\CreatePaymentRequest($token, Str::random(32), $amount_money);
|
2021-08-18 17:23:31 +02:00
|
|
|
|
|
|
|
$body->setAutocomplete(true);
|
|
|
|
$body->setLocationId($this->square_driver->company_gateway->getConfigField('locationId'));
|
|
|
|
$body->setReferenceId(Str::random(16));
|
|
|
|
|
2021-08-19 13:34:45 +02:00
|
|
|
if ($request->shouldUseToken()) {
|
|
|
|
$body->setCustomerId($cgt->gateway_customer_reference);
|
|
|
|
}
|
|
|
|
|
2021-08-18 17:23:31 +02:00
|
|
|
/** @var ApiResponse */
|
|
|
|
$response = $this->square_driver->square->getPaymentsApi()->createPayment($body);
|
|
|
|
|
|
|
|
if ($response->isSuccess()) {
|
2021-08-19 13:34:45 +02:00
|
|
|
if ($request->shouldStoreToken()) {
|
|
|
|
$this->storePaymentMethod($response);
|
|
|
|
}
|
|
|
|
|
2021-08-18 17:23:31 +02:00
|
|
|
return $this->processSuccessfulPayment($response);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->processUnsuccessfulPayment($response);
|
2021-08-14 10:11:45 +02:00
|
|
|
}
|
|
|
|
|
2021-08-19 13:34:45 +02:00
|
|
|
private function storePaymentMethod(ApiResponse $response)
|
|
|
|
{
|
|
|
|
$payment = \json_decode($response->getBody());
|
|
|
|
|
|
|
|
$card = new \Square\Models\Card();
|
|
|
|
$card->setCardholderName($this->square_driver->client->present()->name());
|
|
|
|
$card->setCustomerId($this->findOrCreateClient());
|
|
|
|
$card->setReferenceId(Str::random(8));
|
|
|
|
|
|
|
|
$body = new \Square\Models\CreateCardRequest(Str::random(32), $payment->payment->id, $card);
|
|
|
|
|
|
|
|
/** @var ApiResponse */
|
|
|
|
$api_response = $this->square_driver
|
|
|
|
->square
|
|
|
|
->getCardsApi()
|
|
|
|
->createCard($body);
|
|
|
|
|
|
|
|
if (!$api_response->isSuccess()) {
|
|
|
|
return $this->processUnsuccessfulPayment($api_response);
|
|
|
|
}
|
|
|
|
|
|
|
|
$card = \json_decode($api_response->getBody());
|
|
|
|
|
|
|
|
$cgt = [];
|
|
|
|
$cgt['token'] = $card->card->id;
|
|
|
|
$cgt['payment_method_id'] = GatewayType::CREDIT_CARD;
|
|
|
|
|
|
|
|
$payment_meta = new \stdClass;
|
|
|
|
$payment_meta->exp_month = $card->card->exp_month;
|
|
|
|
$payment_meta->exp_year = $card->card->exp_year;
|
|
|
|
$payment_meta->brand = $card->card->card_brand;
|
|
|
|
$payment_meta->last4 = $card->card->last_4;
|
|
|
|
$payment_meta->type = GatewayType::CREDIT_CARD;
|
|
|
|
|
|
|
|
$cgt['payment_meta'] = $payment_meta;
|
|
|
|
|
|
|
|
$this->square_driver->storeGatewayToken($cgt, [
|
|
|
|
'gateway_customer_reference' => $this->findOrCreateClient(),
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2021-08-18 17:23:31 +02:00
|
|
|
private function processSuccessfulPayment(ApiResponse $response)
|
2021-08-14 10:11:45 +02:00
|
|
|
{
|
2021-08-18 17:23:31 +02:00
|
|
|
$body = json_decode($response->getBody());
|
|
|
|
|
2021-08-14 13:00:52 +02:00
|
|
|
$amount = array_sum(array_column($this->square_driver->payment_hash->invoices(), 'amount')) + $this->square_driver->payment_hash->fee_total;
|
2021-08-14 10:11:45 +02:00
|
|
|
|
|
|
|
$payment_record = [];
|
|
|
|
$payment_record['amount'] = $amount;
|
|
|
|
$payment_record['payment_type'] = PaymentType::CREDIT_CARD_OTHER;
|
|
|
|
$payment_record['gateway_type_id'] = GatewayType::CREDIT_CARD;
|
2021-08-18 17:23:31 +02:00
|
|
|
$payment_record['transaction_reference'] = $body->payment->id;
|
2021-08-14 10:11:45 +02:00
|
|
|
|
2021-08-14 13:00:52 +02:00
|
|
|
$payment = $this->square_driver->createPayment($payment_record, Payment::STATUS_COMPLETED);
|
2021-08-14 10:11:45 +02:00
|
|
|
|
|
|
|
return redirect()->route('client.payments.show', ['payment' => $this->encodePrimaryKey($payment->id)]);
|
|
|
|
}
|
|
|
|
|
2021-08-18 17:23:31 +02:00
|
|
|
private function processUnsuccessfulPayment(ApiResponse $response)
|
2021-08-14 10:11:45 +02:00
|
|
|
{
|
2021-08-19 13:34:45 +02:00
|
|
|
$body = \json_decode($response->getBody());
|
2021-08-14 10:11:45 +02:00
|
|
|
|
|
|
|
$data = [
|
|
|
|
'response' => $response,
|
2021-08-19 13:34:45 +02:00
|
|
|
'error' => $body->errors[0]->detail,
|
2021-08-15 12:39:05 +02:00
|
|
|
'error_code' => '',
|
2021-08-14 10:11:45 +02:00
|
|
|
];
|
|
|
|
|
2021-08-14 13:00:52 +02:00
|
|
|
return $this->square_driver->processUnsuccessfulTransaction($data);
|
2021-08-14 10:11:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2021-08-15 13:14:18 +02:00
|
|
|
private function findOrCreateClient()
|
|
|
|
{
|
|
|
|
$email_address = new \Square\Models\CustomerTextFilter();
|
|
|
|
$email_address->setExact($this->square_driver->client->present()->email());
|
|
|
|
|
|
|
|
$filter = new \Square\Models\CustomerFilter();
|
|
|
|
$filter->setEmailAddress($email_address);
|
|
|
|
|
|
|
|
$query = new \Square\Models\CustomerQuery();
|
|
|
|
$query->setFilter($filter);
|
|
|
|
|
|
|
|
$body = new \Square\Models\SearchCustomersRequest();
|
|
|
|
$body->setQuery($query);
|
|
|
|
|
|
|
|
$api_response = $this->square_driver
|
|
|
|
->init()
|
|
|
|
->square
|
|
|
|
->getCustomersApi()
|
|
|
|
->searchCustomers($body);
|
|
|
|
|
|
|
|
$customers = false;
|
|
|
|
|
|
|
|
if ($api_response->isSuccess()) {
|
|
|
|
$customers = $api_response->getBody();
|
|
|
|
$customers = json_decode($customers);
|
|
|
|
} else {
|
|
|
|
$errors = $api_response->getErrors();
|
|
|
|
}
|
|
|
|
|
2021-08-18 17:07:15 +02:00
|
|
|
if ($customers) {
|
2021-08-15 13:14:18 +02:00
|
|
|
return $customers->customers[0]->id;
|
2021-08-18 17:07:15 +02:00
|
|
|
}
|
2021-08-15 13:14:18 +02:00
|
|
|
|
|
|
|
return $this->createClient();
|
|
|
|
}
|
|
|
|
|
|
|
|
private function createClient()
|
|
|
|
{
|
|
|
|
|
|
|
|
/* Step two - create the customer */
|
|
|
|
$billing_address = new \Square\Models\Address();
|
|
|
|
$billing_address->setAddressLine1($this->square_driver->client->address1);
|
|
|
|
$billing_address->setAddressLine2($this->square_driver->client->address2);
|
|
|
|
$billing_address->setLocality($this->square_driver->client->city);
|
|
|
|
$billing_address->setAdministrativeDistrictLevel1($this->square_driver->client->state);
|
|
|
|
$billing_address->setPostalCode($this->square_driver->client->postal_code);
|
|
|
|
$billing_address->setCountry($this->square_driver->client->country->iso_3166_2);
|
|
|
|
|
|
|
|
$body = new \Square\Models\CreateCustomerRequest();
|
|
|
|
$body->setGivenName($this->square_driver->client->present()->name());
|
|
|
|
$body->setFamilyName('');
|
|
|
|
$body->setEmailAddress($this->square_driver->client->present()->email());
|
|
|
|
$body->setAddress($billing_address);
|
|
|
|
$body->setPhoneNumber($this->square_driver->client->phone);
|
|
|
|
$body->setReferenceId($this->square_driver->client->number);
|
|
|
|
$body->setNote('Created by Invoice Ninja.');
|
2021-08-14 10:11:45 +02:00
|
|
|
|
2021-08-15 13:14:18 +02:00
|
|
|
$api_response = $this->square_driver
|
|
|
|
->init()
|
|
|
|
->square
|
|
|
|
->getCustomersApi()
|
|
|
|
->createCustomer($body);
|
2021-08-14 10:11:45 +02:00
|
|
|
|
2021-08-15 13:14:18 +02:00
|
|
|
if ($api_response->isSuccess()) {
|
|
|
|
$result = $api_response->getResult();
|
|
|
|
return $result->getCustomer()->getId();
|
|
|
|
} else {
|
|
|
|
$errors = $api_response->getErrors();
|
|
|
|
return $this->processUnsuccessfulPayment($errors);
|
|
|
|
}
|
|
|
|
}
|
2021-08-18 17:07:15 +02:00
|
|
|
}
|