mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-10-31 02:57:30 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			348 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			348 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?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\WePay;
 | |
| 
 | |
| use App\Exceptions\PaymentFailed;
 | |
| use App\Models\ClientGatewayToken;
 | |
| use App\Models\GatewayType;
 | |
| use App\Models\Payment;
 | |
| use App\PaymentDrivers\WePayPaymentDriver;
 | |
| use App\PaymentDrivers\WePay\WePayCommon;
 | |
| use App\Utils\Traits\MakesHash;
 | |
| use Illuminate\Http\Request;
 | |
| use Illuminate\Support\Str;
 | |
| 
 | |
| class ACH
 | |
| {
 | |
|     use MakesHash;
 | |
|     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.bank_transfer', $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('payment_bank/persist', [
 | |
|                 'client_id'          => config('ninja.wepay.client_id'),
 | |
|                 'client_secret'      => config('ninja.wepay.client_secret'),
 | |
|                 'payment_bank_id'     => (int)$data['bank_account_id'],
 | |
|             ]);
 | |
|         }
 | |
|         catch(\Exception $e){
 | |
|             throw new PaymentFailed($e->getMessage(), 400);
 | |
|         }
 | |
|         // display the response
 | |
|         // nlog($response);
 | |
|         
 | |
|         if(in_array($response->state, ['new', 'pending', 'authorized'])){
 | |
| 
 | |
|             $this->storePaymentMethod($response, GatewayType::BANK_TRANSFER);
 | |
| 
 | |
|             return redirect()->route('client.payment_methods.index');
 | |
| 
 | |
|         }
 | |
|     
 | |
|         throw new PaymentFailed("There was a problem adding this payment method.", 400);
 | |
|         
 | |
|         /*
 | |
|             {
 | |
|                 "payment_bank_id": 12345,
 | |
|                 "bank_name": "Wells Fargo",
 | |
|                 "account_last_four": "6789",
 | |
|                 "state": "authorized"
 | |
|             }
 | |
| 
 | |
|             state options: new, pending, authorized, disabled.
 | |
|         */    
 | |
|     
 | |
|     }  
 | |
| 
 | |
| 
 | |
| /* If the bank transfer token is PENDING - we need to verify!! */
 | |
| //
 | |
| 
 | |
|     public function verificationView(ClientGatewayToken $token)
 | |
|     {
 | |
|         $this->wepay_payment_driver->init();
 | |
| 
 | |
|         $data = [
 | |
|             'token' => $token,
 | |
|             'gateway' => $this->wepay_payment_driver,
 | |
|         ];
 | |
| 
 | |
|         return render('gateways.wepay.authorize.verify', $data);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|     {
 | |
|     "client_id": 1234,
 | |
|     "client_secret": "b1fc2f68-4d1f-4a",
 | |
|     "payment_bank_id": 12345,
 | |
|     "type": "microdeposits",
 | |
|     "microdeposits": [
 | |
|         8,
 | |
|         12
 | |
|     ]
 | |
|     }
 | |
|      */
 | |
|     public function processVerification(Request $request, ClientGatewayToken $token)
 | |
|     {
 | |
|         $transactions = $request->input('transactions');
 | |
| 
 | |
|         $transformed_transactions = [];
 | |
| 
 | |
|         foreach($transactions as $transaction)
 | |
|             $transformed_transactions[] = (int)$transaction;
 | |
| 
 | |
|         try {
 | |
|         
 | |
|             $response = $this->wepay_payment_driver->wepay->request('payment_bank/verify', [
 | |
|                 'client_id'          => config('ninja.wepay.client_id'),
 | |
|                 'client_secret'      => config('ninja.wepay.client_secret'),
 | |
|                 'payment_bank_id'    => $token->token,
 | |
|                 'type'               => 'microdeposits',
 | |
|                 'microdeposits'      => $transformed_transactions,
 | |
|             ]);
 | |
|         
 | |
|         }
 | |
|         catch(\Exception $e){
 | |
| 
 | |
|             nlog("we pay exception");
 | |
|             nlog($e->getMessage());
 | |
|             
 | |
|             return redirect()->route('client.payment_methods.verification', ['payment_method' => $token->hashed_id, 'method' => GatewayType::BANK_TRANSFER])
 | |
|                       ->with('error', $e->getMessage());
 | |
|         
 | |
|         }
 | |
|         /*
 | |
|         {
 | |
|             "payment_bank_id": 12345,
 | |
|             "bank_name": "Wells Fargo",
 | |
|             "account_last_four": "6789",
 | |
|             "state": "authorized"
 | |
|         }
 | |
|         */
 | |
|        nlog($response);
 | |
| 
 | |
|        //$meta = $token->meta;
 | |
|         if($response->state == "authorized")
 | |
|         {
 | |
|             $meta = $token->meta;
 | |
|             $meta->state = $response->state;
 | |
|             $token->meta = $meta;
 | |
|             $token->save();
 | |
| 
 | |
|             return redirect()->route('client.payment_methods.index');
 | |
| 
 | |
|         }
 | |
|         else{
 | |
|          
 | |
|           return redirect()->route('client.payment_methods.verification', ['payment_method' => $token->hashed_id, 'method' => GatewayType::BANK_TRANSFER])
 | |
|                       ->with('error', ctrans('texts.verification_failed'));   
 | |
|         }
 | |
|     }
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////////////
 | |
|     public function paymentView(array $data)
 | |
|     {
 | |
| 
 | |
|         $data['gateway'] = $this->wepay_payment_driver;
 | |
|         $data['currency'] = $this->wepay_payment_driver->client->getCurrencyCode();
 | |
|         $data['payment_method_id'] = GatewayType::BANK_TRANSFER;
 | |
|         $data['amount'] = $data['total']['amount_with_fee'];
 | |
| 
 | |
|         return render('gateways.wepay.bank_transfer', $data);
 | |
|     }
 | |
| 
 | |
| 
 | |
|     public function paymentResponse($request)
 | |
|     {
 | |
|         // nlog($request->all());
 | |
|         
 | |
|         $token = ClientGatewayToken::find($this->decodePrimaryKey($request->input('source')));
 | |
|         $token_meta = $token->meta;
 | |
| 
 | |
|         if(!property_exists($token_meta, 'state') || $token_meta->state != "authorized")
 | |
|             return redirect()->route('client.payment_methods.verification', ['payment_method' => $token->hashed_id, 'method' => GatewayType::BANK_TRANSFER]);
 | |
| 
 | |
|         $app_fee = (config('ninja.wepay.fee_ach_multiplier') * $this->wepay_payment_driver->payment_hash->data->amount_with_fee) + config('ninja.wepay.fee_fixed');
 | |
| 
 | |
|         try{
 | |
|             $response = $this->wepay_payment_driver->wepay->request('checkout/create', array(
 | |
|                 // 'callback_uri'        => route('payment_webhook', ['company_key' => $this->wepay_payment_driver->company_gateway->company->company_key, 'company_gateway_id' => $this->wepay_payment_driver->company_gateway->hashed_id]),
 | |
|                 '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'      => array(
 | |
|                     'type'            => 'payment_bank',
 | |
|                     'payment_bank'     => array(
 | |
|                         'id'          => $token->token
 | |
|                     )
 | |
|                 )
 | |
|             ));
 | |
|         }
 | |
|         catch(\Exception $e){
 | |
|             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::BANK_TRANSFER);
 | |
|         }
 | |
| 
 | |
|         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);
 | |
|         }
 | |
| 
 | |
|     }
 | |
| 
 | |
|     private function storePaymentMethod($response, $payment_method_id)
 | |
|     {
 | |
| 
 | |
|         $payment_meta = new \stdClass;
 | |
|         $payment_meta->exp_month = (string) '';
 | |
|         $payment_meta->exp_year = (string) '';
 | |
|         $payment_meta->brand = (string) $response->bank_name;
 | |
|         $payment_meta->last4 = (string) $response->account_last_four;
 | |
|         $payment_meta->type = GatewayType::BANK_TRANSFER;
 | |
|         $payment_meta->state = $response->state;
 | |
| 
 | |
|         $data = [
 | |
|             'payment_meta' => $payment_meta,
 | |
|             'token' => $response->payment_bank_id,
 | |
|             'payment_method_id' => $payment_method_id,
 | |
|         ];
 | |
| 
 | |
|         $this->wepay_payment_driver->storeGatewayToken($data);
 | |
| 
 | |
|     }     
 | |
| 
 | |
| 
 | |
|     public function tokenBilling($token, $payment_hash)
 | |
|     {
 | |
|         
 | |
|         $token_meta = $token->meta;
 | |
| 
 | |
|         if(!property_exists($token_meta, 'state') || $token_meta->state != "authorized")
 | |
|             return redirect()->route('client.payment_methods.verification', ['payment_method' => $token->hashed_id, 'method' => GatewayType::BANK_TRANSFER]);
 | |
| 
 | |
|         $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');
 | |
| 
 | |
|         $response = $this->wepay_payment_driver->wepay->request('checkout/create', array(
 | |
|             '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'      => array(
 | |
|                 'type'            => 'payment_bank',
 | |
|                 'payment_bank'     => array(
 | |
|                     'id'          => $token->token
 | |
|                 )
 | |
|             )
 | |
|         ));
 | |
| 
 | |
|                 /* Merge all data and store in the payment hash*/
 | |
|         $state = [
 | |
|             'server_response' => $response,
 | |
|             'payment_hash' => $this->wepay_payment_driver->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::BANK_TRANSFER, 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);
 | |
|         }
 | |
| 
 | |
|     }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| }
 |