mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-11-03 18:57:34 -05:00 
			
		
		
		
	Shift automatically applies the Laravel coding style - which uses the PSR-2 coding style as a base with some minor additions. You may customize the code style applied by adding a [PHP CS Fixer][1] or [PHP CodeSniffer][2] ruleset to your project root. Feel free to use [Shift's Laravel ruleset][3] to help you get started. For more information on customizing the code style applied by Shift, [watch this short video][4]. [1]: https://github.com/FriendsOfPHP/PHP-CS-Fixer [2]: https://github.com/squizlabs/PHP_CodeSniffer [3]: https://gist.github.com/laravel-shift/cab527923ed2a109dda047b97d53c200 [4]: https://laravelshift.com/videos/shift-code-style
		
			
				
	
	
		
			347 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			347 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
/**
 | 
						|
 * Invoice Ninja (https://invoiceninja.com).
 | 
						|
 *
 | 
						|
 * @link https://github.com/invoiceninja/invoiceninja source repository
 | 
						|
 *
 | 
						|
 * @copyright Copyright (c) 2022. 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\PaymentType;
 | 
						|
use App\Models\SystemLog;
 | 
						|
use App\PaymentDrivers\WePay\WePayCommon;
 | 
						|
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;
 | 
						|
 | 
						|
        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');
 | 
						|
 | 
						|
        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);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |