mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-10-31 08:07:33 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			500 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			500 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * Invoice Ninja (https://invoiceninja.com)
 | |
|  *
 | |
|  * @link https://github.com/invoiceninja/invoiceninja source repository
 | |
|  *
 | |
|  * @copyright Copyright (c) 2019. Invoice Ninja LLC (https://invoiceninja.com)
 | |
|  *
 | |
|  * @license https://opensource.org/licenses/AAL
 | |
|  */
 | |
| 
 | |
| namespace App\PaymentDrivers;
 | |
| 
 | |
| use App\Events\Payment\PaymentWasCreated;
 | |
| use App\Factory\PaymentFactory;
 | |
| use App\Models\ClientGatewayToken;
 | |
| use App\Models\GatewayType;
 | |
| use App\Models\Invoice;
 | |
| use App\Models\Payment;
 | |
| use App\Models\PaymentType;
 | |
| use App\Utils\Traits\MakesHash;
 | |
| use Illuminate\Http\Request;
 | |
| use Illuminate\Support\Carbon;
 | |
| use Stripe\PaymentIntent;
 | |
| use Stripe\SetupIntent;
 | |
| use Stripe\Stripe;
 | |
| 
 | |
| class StripePaymentDriver extends BasePaymentDriver
 | |
| {
 | |
|   use MakesHash;
 | |
| 
 | |
| 	protected $refundable = true;
 | |
| 
 | |
| 	protected $token_billing = true;
 | |
| 
 | |
|   protected $can_authorise_credit_card = true;
 | |
| 
 | |
| 	protected $customer_reference = 'customerReferenceParam';
 | |
| 
 | |
| 	/**
 | |
| 	 * Methods in this class are divided into
 | |
| 	 * two separate streams
 | |
| 	 * 
 | |
| 	 * 1. Omnipay Specific
 | |
| 	 * 2. Stripe Specific
 | |
| 	 * 
 | |
| 	 * Our Stripe integration is deeper than 
 | |
| 	 * other gateways and therefore
 | |
| 	 * relies on direct calls to the API
 | |
| 	 */
 | |
| 	/************************************** Stripe API methods **********************************************************/
 | |
| 
 | |
|     /**
 | |
|      * Initializes the Stripe API
 | |
|      * @return void
 | |
|      */
 | |
| 	public function init() :void
 | |
| 	{
 | |
|         Stripe::setApiKey($this->company_gateway->getConfigField('23_apiKey'));
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Returns the gateway types
 | |
| 	 */
 | |
|     public function gatewayTypes() :array
 | |
|     {
 | |
|         $types = [
 | |
|             GatewayType::CREDIT_CARD,
 | |
|             //GatewayType::TOKEN,
 | |
|         ];
 | |
|         
 | |
|         if($this->company_gateway->getSofortEnabled() && $this->invitation && $this->client() && isset($this->client()->country) && in_array($this->client()->country, ['AUT', 'BEL', 'DEU', 'ITA', 'NLD', 'ESP']))
 | |
|             $types[] = GatewayType::SOFORT;
 | |
| 
 | |
| 	    if($this->company_gateway->getAchEnabled())
 | |
| 	    	$types[] = GatewayType::BANK_TRANSFER;
 | |
| 
 | |
|         if ($this->company_gateway->getSepaEnabled()) 
 | |
|             $types[] = GatewayType::SEPA;
 | |
|         
 | |
|         if ($this->company_gateway->getBitcoinEnabled()) 
 | |
|             $types[] = GatewayType::BITCOIN;
 | |
|         
 | |
|         if ($this->company_gateway->getAlipayEnabled()) 
 | |
|             $types[] = GatewayType::ALIPAY;
 | |
|         
 | |
|         if ($this->company_gateway->getApplePayEnabled()) 
 | |
|             $types[] = GatewayType::APPLE_PAY;
 | |
|             
 | |
| 
 | |
|         return $types;
 | |
| 
 | |
|     }
 | |
| 
 | |
|     public function viewForType($gateway_type_id)
 | |
|     {
 | |
|     	switch ($gateway_type_id) {
 | |
|     		case GatewayType::CREDIT_CARD:
 | |
|     			return 'portal.default.gateways.stripe.credit_card';
 | |
|     			break;
 | |
|     		case GatewayType::TOKEN:
 | |
|     			return 'portal.default.gateways.stripe.credit_card';
 | |
|     			break;
 | |
|     		case GatewayType::SOFORT:
 | |
|     			return 'portal.default.gateways.stripe.sofort';
 | |
|     			break;
 | |
|     		case GatewayType::BANK_TRANSFER:
 | |
|     			return 'portal.default.gateways.stripe.ach';
 | |
|     			break;
 | |
|     		case GatewayType::SEPA:
 | |
|     			return 'portal.default.gateways.stripe.sepa';
 | |
|     			break;
 | |
|     		case GatewayType::BITCOIN:
 | |
|     			return 'portal.default.gateways.stripe.other';
 | |
|     			break;
 | |
|     		case GatewayType::ALIPAY:
 | |
|     			return 'portal.default.gateways.stripe.other';
 | |
|     			break;
 | |
|     		case GatewayType::APPLE_PAY:
 | |
|     			return 'portal.default.gateways.stripe.other';
 | |
|     			break;
 | |
| 
 | |
|     		default:
 | |
|     			# code...
 | |
|     			break;
 | |
|     	}
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 
 | |
|      * Authorises a credit card for future use
 | |
|      * @param  array  $data Array of variables needed for the view
 | |
|      * 
 | |
|      * @return view       The gateway specific partial to be rendered
 | |
|      * 
 | |
|      */
 | |
|     public function authorizeCreditCardView(array $data)
 | |
|     {
 | |
| 
 | |
|         $intent['intent'] = $this->getSetupIntent();
 | |
| 
 | |
|         return view('portal.default.gateways.stripe.add_credit_card', array_merge($data, $intent));
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Processes the gateway response for credti card authorization
 | |
|      * @param  Request $request The returning request object
 | |
|      * @return view          Returns the user to payment methods screen.
 | |
|      */
 | |
|     public function authorizeCreditCardResponse($request)
 | |
|     {
 | |
| 
 | |
|         $server_response = json_decode($request->input('gateway_response'));
 | |
| 
 | |
|         $gateway_id = $request->input('gateway_id');
 | |
|         $gateway_type_id = $request->input('gateway_type_id');
 | |
|         $is_default = $request->input('is_default');
 | |
| 
 | |
|         $payment_method = $server_response->payment_method;
 | |
| 
 | |
|         $customer = $this->findOrCreateCustomer();
 | |
| 
 | |
|         $this->init();
 | |
|         $stripe_payment_method = \Stripe\PaymentMethod::retrieve($payment_method);
 | |
|         $stripe_payment_method_obj = $stripe_payment_method->jsonSerialize();
 | |
|         $stripe_payment_method->attach(['customer' => $customer->id]);
 | |
| 
 | |
|         $payment_meta = new \stdClass;
 | |
| 
 | |
|         if($stripe_payment_method_obj['type'] == 'card') {
 | |
|             $payment_meta->exp_month = $stripe_payment_method_obj['card']['exp_month'];
 | |
|             $payment_meta->exp_year = $stripe_payment_method_obj['card']['exp_year'];
 | |
|             $payment_meta->brand = $stripe_payment_method_obj['card']['brand'];
 | |
|             $payment_meta->last4 = $stripe_payment_method_obj['card']['last4'];
 | |
|             $payment_meta->type = $stripe_payment_method_obj['type'];
 | |
|         }
 | |
| 
 | |
|         $cgt = new ClientGatewayToken;
 | |
|         $cgt->company_id = $this->client->company->id;
 | |
|         $cgt->client_id = $this->client->id;
 | |
|         $cgt->token = $payment_method;
 | |
|         $cgt->company_gateway_id = $this->company_gateway->id;
 | |
|         $cgt->gateway_type_id = $gateway_type_id;
 | |
|         $cgt->gateway_customer_reference = $customer->id;
 | |
|         $cgt->meta = $payment_meta;
 | |
|         $cgt->save();
 | |
| 
 | |
|         if($is_default == 'true' || $this->client->gateway_tokens->count() == 1)
 | |
|         {
 | |
|             $this->client->gateway_tokens()->update(['is_default'=>0]);
 | |
| 
 | |
|             $cgt->is_default = 1;
 | |
|             $cgt->save();
 | |
|         }
 | |
| 
 | |
|         return redirect()->route('client.payment_methods.index');
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Processes the payment with this gateway
 | |
|      *             
 | |
|      * @var         invoices
 | |
|      * @var         amount
 | |
|      * @var         fee
 | |
|      * @var         amount_with_fee
 | |
|      * @var         token
 | |
|      * @var         payment_method_id
 | |
|      * @var         hashed_ids
 | |
|      * 
 | |
|      * @param  array  $data variables required to build payment page
 | |
|      * @return view   Gateway and payment method specific view
 | |
|      */
 | |
|     public function processPaymentView(array $data)
 | |
|     {
 | |
|         $payment_intent_data = [
 | |
|             'amount' => $this->convertToStripeAmount($data['amount_with_fee'], $this->client->currency->precision),
 | |
|             'currency' => $this->client->getCurrencyCode(),
 | |
|             'customer' => $this->findOrCreateCustomer(),
 | |
|             'description' => $data['invoices']->pluck('id'), //todo more meaningful description here:
 | |
|         ];
 | |
| 
 | |
|         if($data['token'])
 | |
|             $payment_intent_data['payment_method'] = $data['token']->token;
 | |
|         else{
 | |
|             $payment_intent_data['setup_future_usage']  = 'off_session';
 | |
| //            $payment_intent_data['save_payment_method'] = true;
 | |
| //            $payment_intent_data['confirm'] = true;
 | |
|         }
 | |
| 
 | |
| 
 | |
|         $data['intent'] = $this->createPaymentIntent($payment_intent_data);
 | |
| 
 | |
|         $data['gateway'] = $this;
 | |
| 
 | |
|         return view($this->viewForType($data['payment_method_id']), $data);
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Payment Intent Reponse looks like this
 | |
|       +"id": "pi_1FMR7JKmol8YQE9DuC4zMeN3"
 | |
|       +"object": "payment_intent"
 | |
|       +"allowed_source_types": array:1 [▼
 | |
|         0 => "card"
 | |
|       ]
 | |
|       +"amount": 2372484
 | |
|       +"canceled_at": null
 | |
|       +"cancellation_reason": null
 | |
|       +"capture_method": "automatic"
 | |
|       +"client_secret": "pi_1FMR7JKmol8YQE9DuC4zMeN3_secret_J3yseWJG6uV0MmsrAT1FlUklV"
 | |
|       +"confirmation_method": "automatic"
 | |
|       +"created": 1569381877
 | |
|       +"currency": "usd"
 | |
|       +"description": "[3]"
 | |
|       +"last_payment_error": null
 | |
|       +"livemode": false
 | |
|       +"next_action": null
 | |
|       +"next_source_action": null
 | |
|       +"payment_method": "pm_1FMR7ZKmol8YQE9DQWqPuyke"
 | |
|       +"payment_method_types": array:1 [▶]
 | |
|       +"receipt_email": null
 | |
|       +"setup_future_usage": "off_session"
 | |
|       +"shipping": null
 | |
|       +"source": null
 | |
|       +"status": "succeeded"
 | |
|     */
 | |
|     public function processPaymentResponse($request)
 | |
|     {
 | |
|         $server_response = json_decode($request->input('gateway_response'));
 | |
| 
 | |
|         $payment_method = $server_response->payment_method;
 | |
|         $payment_status = $server_response->status;
 | |
|         $save_card = $request->input('store_card');
 | |
| 
 | |
|         $gateway_type_id = $request->input('payment_method_id');
 | |
|         $hashed_ids = $request->input('hashed_ids');
 | |
|         $invoices = Invoice::whereIn('id', $this->transformKeys(explode(",",$hashed_ids)))
 | |
|                                 ->whereClientId($this->client->id)
 | |
|                                 ->get();
 | |
|         /**
 | |
|          * Potential statuses that can be returned
 | |
|          * 
 | |
|          * requires_action
 | |
|          * processing
 | |
|          * canceled
 | |
|          * requires_action
 | |
|          * requires_confirmation
 | |
|          * requires_payment_method 
 | |
|          * 
 | |
|          */
 | |
|         
 | |
|         if($this->getContact()){
 | |
|           $client_contact = $this->getContact();
 | |
|         }
 | |
|         else{
 | |
|           $client_contact = $invoices->first()->invitations->first()->contact;
 | |
|         }
 | |
| 
 | |
|         $this->init();
 | |
|         $payment_intent = \Stripe\PaymentIntent::retrieve($server_response->id);
 | |
|         $customer = $payment_intent->customer;
 | |
| 
 | |
|         if($payment_status == 'succeeded')
 | |
|         {
 | |
|             $this->init();
 | |
|             $stripe_payment_method = \Stripe\PaymentMethod::retrieve($payment_method);
 | |
|             $stripe_payment_method_obj = $stripe_payment_method->jsonSerialize();
 | |
| 
 | |
|             $payment_meta = new \stdClass;
 | |
| 
 | |
|             if($stripe_payment_method_obj['type'] == 'card') {
 | |
|                 $payment_meta->exp_month = $stripe_payment_method_obj['card']['exp_month'];
 | |
|                 $payment_meta->exp_year = $stripe_payment_method_obj['card']['exp_year'];
 | |
|                 $payment_meta->brand = $stripe_payment_method_obj['card']['brand'];
 | |
|                 $payment_meta->last4 = $stripe_payment_method_obj['card']['last4'];
 | |
|                 $payment_meta->type = $stripe_payment_method_obj['type'];
 | |
| 
 | |
|                 $payment_type = PaymentType::parseCardType($stripe_payment_method_obj['card']['brand']);
 | |
|             }
 | |
| 
 | |
|             if($save_card == 'true')
 | |
|             {
 | |
| 
 | |
|                 $stripe_payment_method->attach(['customer' => $customer]);
 | |
|                 
 | |
|                 $cgt = new ClientGatewayToken;
 | |
|                 $cgt->company_id = $this->client->company->id;
 | |
|                 $cgt->client_id = $this->client->id;
 | |
|                 $cgt->token = $payment_method;
 | |
|                 $cgt->company_gateway_id = $this->company_gateway->id;
 | |
|                 $cgt->gateway_type_id = $gateway_type_id;
 | |
|                 $cgt->gateway_customer_reference = $customer;
 | |
|                 $cgt->meta = $payment_meta;
 | |
|                 $cgt->save();
 | |
| 
 | |
|                 if($this->client->gateway_tokens->count() == 1)
 | |
|                 {
 | |
|                     $this->client->gateway_tokens()->update(['is_default'=>0]);
 | |
| 
 | |
|                     $cgt->is_default = 1;
 | |
|                     $cgt->save();
 | |
|                 }  
 | |
|             }
 | |
|         
 | |
|             //todo need to fix this to support payment types other than credit card.... sepa etc etc
 | |
|             if(!$payment_type)
 | |
|               $payment_type = PaymentType::CREDIT_CARD_OTHER;
 | |
| 
 | |
| 
 | |
|             $data = [
 | |
|               'payment_method' => $payment_method,
 | |
|               'payment_type' => $payment_type,
 | |
|               'amount' => $server_response->amount,
 | |
|             ];
 | |
| 
 | |
|             /* Create payment*/
 | |
|             $payment = $this->createPayment($data);
 | |
| 
 | |
|             /* Link invoices to payment*/
 | |
|             $this->attachInvoices($payment, $hashed_ids);
 | |
| 
 | |
|             event(new PaymentWasCreated($payment));
 | |
| 
 | |
|             return redirect()->route('client.payments.show', ['payment' => $this->encodePrimaryKey($payment->id)]);
 | |
| 
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|           /*Fail and log*/
 | |
| 
 | |
|           $log = [
 | |
|             'server_response' => $server_response,
 | |
|             'invoices' => $invoices,
 | |
|           ];
 | |
| 
 | |
|           $this->sysLog($log);
 | |
| 
 | |
|           throw new Exception("Failed to process payment", 1);
 | |
|           
 | |
|         }
 | |
| 
 | |
|     }
 | |
| 
 | |
|     public function createPayment($data) :Payment
 | |
|     {
 | |
| 
 | |
|         $payment = parent::createPayment($data);
 | |
| 
 | |
|         $client_contact = $this->getContact();
 | |
|         $client_contact_id = $client_contact ? $client_contact->id : null;
 | |
| 
 | |
|         $payment->amount = $this->convertFromStripeAmount($data['amount'], $this->client->currency->precision);
 | |
|         $payment->payment_type_id = $data['payment_type'];
 | |
|         $payment->transaction_reference = $data['payment_method'];
 | |
|         $payment->client_contact_id = $client_contact_id;
 | |
|         $payment->save();
 | |
| 
 | |
|         return $payment;
 | |
| 
 | |
|     }
 | |
| 
 | |
|     private function convertFromStripeAmount($amount, $precision)
 | |
|     {
 | |
|       return $amount / pow(10, $precision);
 | |
|     }
 | |
| 
 | |
|     private function convertToStripeAmount($amount, $precision)
 | |
|     {
 | |
|       return $amount * pow(10, $precision);
 | |
|     }
 | |
|     /**
 | |
|      * Creates a new String Payment Intent
 | |
|      * 
 | |
|      * @param  array $data The data array to be passed to Stripe
 | |
|      * @return PaymentIntent       The Stripe payment intent object
 | |
|      */
 | |
|     public function createPaymentIntent($data) :?\Stripe\PaymentIntent
 | |
|     {
 | |
| 
 | |
|         $this->init();
 | |
| 
 | |
|         return PaymentIntent::create($data);
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns a setup intent that allows the user 
 | |
|      * to enter card details without initiating a transaction.
 | |
|      *
 | |
|      * @return \Stripe\SetupIntent
 | |
|      */
 | |
|     public function getSetupIntent() :\Stripe\SetupIntent
 | |
|     {
 | |
| 
 | |
|         $this->init();
 | |
| 
 | |
|         return SetupIntent::create();
 | |
| 
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /**
 | |
|      * Returns the Stripe publishable key
 | |
|      * @return NULL|string The stripe publishable key
 | |
|      */
 | |
|     public function getPublishableKey() :?string
 | |
|     {
 | |
| 
 | |
|         return $this->company_gateway->getPublishableKey();
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Finds or creates a Stripe Customer object
 | |
|      * 
 | |
|      * @return NULL|\Stripe\Customer A Stripe customer object
 | |
|      */
 | |
|     public function findOrCreateCustomer() :?\Stripe\Customer
 | |
|     { 
 | |
| 
 | |
|         $customer = null;
 | |
| 
 | |
|         $this->init();
 | |
| 
 | |
|         $client_gateway_token = ClientGatewayToken::whereClientId($this->client->id)->whereCompanyGatewayId($this->company_gateway->id)->first();
 | |
| 
 | |
|         if($client_gateway_token && $client_gateway_token->gateway_customer_reference){
 | |
| 
 | |
|             $customer = \Stripe\Customer::retrieve($client_gateway_token->gateway_customer_reference);
 | |
| 
 | |
|         }
 | |
|         else{
 | |
| 
 | |
|             $data['name'] = $this->client->present()->name();
 | |
|             $data['phone'] = $this->client->present()->phone();
 | |
| 
 | |
|             if(filter_var($this->client->present()->email(), FILTER_VALIDATE_EMAIL))
 | |
|                 $data['email'] = $this->client->present()->email();
 | |
| 
 | |
|             $customer = \Stripe\Customer::create($data);
 | |
| 
 | |
|         }
 | |
| 
 | |
|         if(!$customer)
 | |
|             throw new Exception('Unable to create gateway customer');
 | |
| 
 | |
|         return $customer;
 | |
|     }
 | |
| 
 | |
| 
 | |
| 	/************************************** Omnipay API methods **********************************************************/
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| } |