mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2025-10-24 06:49:23 -04:00
347 lines
14 KiB
PHP
347 lines
14 KiB
PHP
<?php
|
|
|
|
/**
|
|
* Invoice Ninja (https://invoiceninja.com).
|
|
*
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
*
|
|
* @copyright Copyright (c) 2023. Invoice Ninja LLC (https://invoiceninja.com)
|
|
*
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
|
*/
|
|
|
|
namespace App\PaymentDrivers\WePay;
|
|
|
|
use App\Exceptions\PaymentFailed;
|
|
use App\Http\Requests\ClientPortal\Payments\PaymentResponseRequest;
|
|
use App\Jobs\Util\SystemLogger;
|
|
use App\Models\GatewayType;
|
|
use App\Models\Payment;
|
|
use App\Models\SystemLog;
|
|
use App\PaymentDrivers\WePayPaymentDriver;
|
|
use Illuminate\Support\Str;
|
|
|
|
class CreditCard
|
|
{
|
|
use WePayCommon;
|
|
|
|
public $wepay_payment_driver;
|
|
|
|
public function __construct(WePayPaymentDriver $wepay_payment_driver)
|
|
{
|
|
$this->wepay_payment_driver = $wepay_payment_driver;
|
|
}
|
|
|
|
public function authorizeView($data)
|
|
{
|
|
$data['gateway'] = $this->wepay_payment_driver;
|
|
$data['country_code'] = $this->wepay_payment_driver?->client?->country ? $this->wepay_payment_driver->client->country->iso_3166_2 : $this->wepay_payment_driver->company_gateway->company->country()->iso_3166_2;
|
|
|
|
return render('gateways.wepay.authorize.authorize', $data);
|
|
}
|
|
|
|
public function authorizeResponse($request)
|
|
{
|
|
//https://developer.wepay.com/api/api-calls/credit_card#authorize
|
|
$data = $request->all();
|
|
// authorize the credit card
|
|
|
|
// nlog($data);
|
|
/*
|
|
'_token' => '1Fk5CRj34up5ntKPvrFyMIAJhDdUNF3boqT3iIN3',
|
|
'company_gateway_id' => '39',
|
|
'payment_method_id' => '1',
|
|
'gateway_response' => NULL,
|
|
'is_default' => NULL,
|
|
'credit_card_id' => '180642154638',
|
|
'q' => '/client/payment_methods',
|
|
'method' => '1',
|
|
*/
|
|
|
|
try {
|
|
$response = $this->wepay_payment_driver->wepay->request('credit_card/authorize', [
|
|
'client_id' => config('ninja.wepay.client_id'),
|
|
'client_secret' => config('ninja.wepay.client_secret'),
|
|
'credit_card_id' => (int) $data['credit_card_id'],
|
|
]);
|
|
} catch (\Exception $e) {
|
|
return $this->wepay_payment_driver->processInternallyFailedPayment($this->wepay_payment_driver, $e);
|
|
}
|
|
// display the response
|
|
// nlog($response);
|
|
|
|
if (in_array($response->state, ['new', 'authorized'])) {
|
|
$this->storePaymentMethod($response, GatewayType::CREDIT_CARD);
|
|
|
|
return redirect()->route('client.payment_methods.index');
|
|
}
|
|
|
|
throw new PaymentFailed('There was a problem adding this payment method.', 400);
|
|
/*
|
|
[credit_card_id] => 348084962473
|
|
[credit_card_name] => Visa xxxxxx4018
|
|
[state] => authorized
|
|
[user_name] => Joey Diaz
|
|
[email] => user@example.com
|
|
[create_time] => 1623798172
|
|
[expiration_month] => 10
|
|
[expiration_year] => 2023
|
|
[last_four] => 4018
|
|
[input_source] => card_keyed
|
|
[virtual_terminal_mode] => none
|
|
[card_on_file] =>
|
|
[recurring] =>
|
|
[cvv_provided] => 1
|
|
[auto_update] =>
|
|
*/
|
|
}
|
|
|
|
public function paymentView(array $data)
|
|
{
|
|
$data['gateway'] = $this->wepay_payment_driver;
|
|
$data['description'] = ctrans('texts.invoices').': '.collect($data['invoices'])->pluck('invoice_number');
|
|
$data['country_code'] = $this->wepay_payment_driver?->client?->country ? $this->wepay_payment_driver->client->country->iso_3166_2 : $this->wepay_payment_driver->company_gateway->company()->iso_3166_2;
|
|
|
|
return render('gateways.wepay.credit_card.pay', $data);
|
|
}
|
|
|
|
public function paymentResponse(PaymentResponseRequest $request)
|
|
{
|
|
nlog('payment response');
|
|
|
|
//it could be an existing token or a new credit_card_id that needs to be converted into a wepay token
|
|
if ($request->has('credit_card_id') && $request->input('credit_card_id')) {
|
|
nlog('authorize the card first!');
|
|
|
|
try {
|
|
$response = $this->wepay_payment_driver->wepay->request('credit_card/authorize', [
|
|
'client_id' => config('ninja.wepay.client_id'),
|
|
'client_secret' => config('ninja.wepay.client_secret'),
|
|
'credit_card_id' => (int) $request->input('credit_card_id'),
|
|
]);
|
|
} catch (\Exception $e) {
|
|
return $this->wepay_payment_driver->processInternallyFailedPayment($this->wepay_payment_driver, $e);
|
|
}
|
|
|
|
$credit_card_id = (int) $response->credit_card_id;
|
|
|
|
if (in_array($response->state, ['new', 'authorized']) && boolval($request->input('store_card'))) {
|
|
$this->storePaymentMethod($response, GatewayType::CREDIT_CARD);
|
|
}
|
|
} else {
|
|
$credit_card_id = (int) $request->input('token');
|
|
}
|
|
|
|
// USD, CAD, and GBP.
|
|
// nlog($request->all());
|
|
|
|
$app_fee = (config('ninja.wepay.fee_cc_multiplier') * $this->wepay_payment_driver->payment_hash->data->amount_with_fee) + config('ninja.wepay.fee_fixed');
|
|
// charge the credit card
|
|
|
|
try {
|
|
$response = $this->wepay_payment_driver->wepay->request('checkout/create', [
|
|
'unique_id' => Str::random(40),
|
|
'account_id' => $this->wepay_payment_driver->company_gateway->getConfigField('accountId'),
|
|
'amount' => $this->wepay_payment_driver->payment_hash->data->amount_with_fee,
|
|
'currency' => $this->wepay_payment_driver->client->getCurrencyCode(),
|
|
'short_description' => 'Goods and services',
|
|
'type' => 'goods',
|
|
'fee' => [
|
|
'fee_payer' => config('ninja.wepay.fee_payer'),
|
|
'app_fee' => $app_fee,
|
|
],
|
|
'payment_method' => [
|
|
'type' => 'credit_card',
|
|
'credit_card' => [
|
|
'id' => $credit_card_id,
|
|
],
|
|
],
|
|
]);
|
|
} catch (\Exception $e) {
|
|
$this->wepay_payment_driver->sendFailureMail($e->getMessage());
|
|
|
|
$message = [
|
|
'server_response' => $e->getMessage(),
|
|
'data' => $this->wepay_payment_driver->payment_hash->data,
|
|
];
|
|
|
|
SystemLogger::dispatch(
|
|
$e->getMessage(),
|
|
SystemLog::CATEGORY_GATEWAY_RESPONSE,
|
|
SystemLog::EVENT_GATEWAY_FAILURE,
|
|
SystemLog::TYPE_WEPAY,
|
|
$this->wepay_payment_driver->client,
|
|
$this->wepay_payment_driver->client->company,
|
|
);
|
|
|
|
throw new PaymentFailed($e->getMessage(), 500);
|
|
}
|
|
|
|
/* Merge all data and store in the payment hash*/
|
|
$state = [
|
|
'server_response' => $response,
|
|
'payment_hash' => $request->payment_hash,
|
|
];
|
|
|
|
$state = array_merge($state, $request->all());
|
|
$this->wepay_payment_driver->payment_hash->data = array_merge((array) $this->wepay_payment_driver->payment_hash->data, $state);
|
|
$this->wepay_payment_driver->payment_hash->save();
|
|
|
|
if (in_array($response->state, ['authorized', 'captured'])) {
|
|
//success
|
|
nlog('success');
|
|
$payment_status = $response->state == 'authorized' ? Payment::STATUS_COMPLETED : Payment::STATUS_PENDING;
|
|
|
|
return $this->processSuccessfulPayment($response, $payment_status, GatewayType::CREDIT_CARD);
|
|
}
|
|
|
|
if (in_array($response->state, ['released', 'cancelled', 'failed', 'expired'])) {
|
|
//some type of failure
|
|
nlog('failure');
|
|
|
|
$payment_status = $response->state == 'cancelled' ? Payment::STATUS_CANCELLED : Payment::STATUS_FAILED;
|
|
|
|
$this->processUnSuccessfulPayment($response, $payment_status);
|
|
}
|
|
}
|
|
|
|
/*
|
|
new The checkout was created by the application. This state typically indicates that checkouts created in WePay's hosted checkout flow are waiting for the payer to submit their information.
|
|
authorized The payer entered their payment info and confirmed the payment on WePay. WePay has successfully charged the card.
|
|
captured The payment has been reserved from the payer.
|
|
released The payment has been credited to the payee account. Note that the released state may be active although there are active partial refunds or partial chargebacks.
|
|
cancelled The payment has been cancelled by the payer, payee, or application.
|
|
refunded The payment was captured and then refunded by the payer, payee, or application. The payment has been debited from the payee account.
|
|
charged back The payment has been charged back by the payer and the payment has been debited from the payee account.
|
|
failed The payment has failed.
|
|
expired Checkouts expire if they remain in the new state for more than 30 minutes (e.g., they have been abandoned).
|
|
*/
|
|
|
|
/*
|
|
https://developer.wepay.com/api/api-calls/checkout
|
|
{
|
|
"checkout_id": 649945633,
|
|
"account_id": 1548718026,
|
|
"type": "donation",
|
|
"short_description": "test checkout",
|
|
"currency": "USD",
|
|
"amount": 20,
|
|
"state": "authorized",
|
|
"soft_descriptor": "WPY*Wolverine",
|
|
"auto_release": true,
|
|
"create_time": 1463589958,
|
|
"gross": 20.88,
|
|
"reference_id": null,
|
|
"callback_uri": null,
|
|
"long_description": null,
|
|
"delivery_type": null,
|
|
"initiated_by": "merchant",
|
|
"in_review": false,
|
|
"fee": {
|
|
"app_fee": 0,
|
|
"processing_fee": 0.88,
|
|
"fee_payer": "payer"
|
|
},
|
|
"chargeback": {
|
|
"amount_charged_back": 0,
|
|
"dispute_uri": null
|
|
},
|
|
"refund": {
|
|
"amount_refunded": 0,
|
|
"refund_reason": null
|
|
},
|
|
"payment_method": {
|
|
"type": "credit_card",
|
|
"credit_card": {
|
|
"id": 1684847614,
|
|
"data": {
|
|
"emv_receipt": null,
|
|
"signature_url": null
|
|
},
|
|
"auto_release": false
|
|
}
|
|
},
|
|
"hosted_checkout": null,
|
|
"payer": {
|
|
"email": "test@example.com",
|
|
"name": "Mr Smith",
|
|
"home_address": null
|
|
},
|
|
"npo_information": null,
|
|
"payment_error": null
|
|
}
|
|
*/
|
|
|
|
private function storePaymentMethod($response, $payment_method_id)
|
|
{
|
|
nlog('storing card');
|
|
|
|
$payment_meta = new \stdClass();
|
|
$payment_meta->exp_month = (string) $response->expiration_month;
|
|
$payment_meta->exp_year = (string) $response->expiration_year;
|
|
$payment_meta->brand = (string) $response->credit_card_name;
|
|
$payment_meta->last4 = (string) $response->last_four;
|
|
$payment_meta->type = GatewayType::CREDIT_CARD;
|
|
|
|
$data = [
|
|
'payment_meta' => $payment_meta,
|
|
'token' => $response->credit_card_id,
|
|
'payment_method_id' => $payment_method_id,
|
|
];
|
|
|
|
$this->wepay_payment_driver->storeGatewayToken($data);
|
|
}
|
|
|
|
public function tokenBilling($cgt, $payment_hash)
|
|
{
|
|
$amount = array_sum(array_column($this->wepay_payment_driver->payment_hash->invoices(), 'amount')) + $this->wepay_payment_driver->payment_hash->fee_total;
|
|
|
|
$app_fee = (config('ninja.wepay.fee_cc_multiplier') * $amount) + config('ninja.wepay.fee_fixed');
|
|
// charge the credit card
|
|
$response = $this->wepay_payment_driver->wepay->request('checkout/create', [
|
|
'unique_id' => Str::random(40),
|
|
'account_id' => $this->wepay_payment_driver->company_gateway->getConfigField('accountId'),
|
|
'amount' => $amount,
|
|
'currency' => $this->wepay_payment_driver->client->getCurrencyCode(),
|
|
'short_description' => 'Goods and services',
|
|
'type' => 'goods',
|
|
'fee' => [
|
|
'fee_payer' => config('ninja.wepay.fee_payer'),
|
|
'app_fee' => $app_fee,
|
|
],
|
|
'payment_method' => [
|
|
'type' => 'credit_card',
|
|
'credit_card' => [
|
|
'id' => $cgt->token,
|
|
],
|
|
],
|
|
]);
|
|
|
|
/* Merge all data and store in the payment hash*/
|
|
$state = [
|
|
'server_response' => $response,
|
|
'payment_hash' => $payment_hash,
|
|
];
|
|
|
|
$this->wepay_payment_driver->payment_hash->data = array_merge((array) $this->wepay_payment_driver->payment_hash->data, $state);
|
|
$this->wepay_payment_driver->payment_hash->save();
|
|
|
|
if (in_array($response->state, ['authorized', 'captured'])) {
|
|
//success
|
|
nlog('success');
|
|
$payment_status = $response->state == 'authorized' ? Payment::STATUS_COMPLETED : Payment::STATUS_PENDING;
|
|
|
|
return $this->processSuccessfulPayment($response, $payment_status, GatewayType::CREDIT_CARD, true);
|
|
}
|
|
|
|
if (in_array($response->state, ['released', 'cancelled', 'failed', 'expired'])) {
|
|
//some type of failure
|
|
nlog('failure');
|
|
|
|
$payment_status = $response->state == 'cancelled' ? Payment::STATUS_CANCELLED : Payment::STATUS_FAILED;
|
|
|
|
$this->processUnSuccessfulPayment($response, $payment_status);
|
|
}
|
|
}
|
|
}
|