mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-10-31 14:47:31 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			334 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			334 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| /**
 | |
|  * Invoice Ninja (https://invoiceninja.com).
 | |
|  *
 | |
|  * @link https://github.com/invoiceninja/invoiceninja source repository
 | |
|  *
 | |
|  * @copyright Copyright (c) 2024. Invoice Ninja LLC (https://invoiceninja.com)
 | |
|  *
 | |
|  * @license https://www.elastic.co/licensing/elastic-license
 | |
|  */
 | |
| 
 | |
| namespace App\PaymentDrivers\GoCardless;
 | |
| 
 | |
| use App\Exceptions\PaymentFailed;
 | |
| use App\Http\Requests\ClientPortal\Payments\PaymentResponseRequest;
 | |
| use App\Jobs\Mail\PaymentFailureMailer;
 | |
| use App\Jobs\Util\SystemLogger;
 | |
| use App\Models\GatewayType;
 | |
| use App\Models\Invoice;
 | |
| use App\Models\Payment;
 | |
| use App\Models\PaymentType;
 | |
| use App\Models\SystemLog;
 | |
| use App\PaymentDrivers\Common\MethodInterface;
 | |
| use App\PaymentDrivers\GoCardlessPaymentDriver;
 | |
| use App\Utils\Traits\MakesHash;
 | |
| use Illuminate\Http\RedirectResponse;
 | |
| use Illuminate\Http\Request;
 | |
| use Illuminate\Routing\Redirector;
 | |
| use Illuminate\View\View;
 | |
| 
 | |
| class DirectDebit implements MethodInterface
 | |
| {
 | |
|     use MakesHash;
 | |
| 
 | |
|     protected GoCardlessPaymentDriver $go_cardless;
 | |
| 
 | |
|     public function __construct(GoCardlessPaymentDriver $go_cardless)
 | |
|     {
 | |
|         $this->go_cardless = $go_cardless;
 | |
| 
 | |
|         $this->go_cardless->init();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Handle authorization for Direct Debit.
 | |
|      *
 | |
|      * @param array $data
 | |
|      * @return Redirector|RedirectResponse|void
 | |
|      */
 | |
|     public function authorizeView(array $data)
 | |
|     {
 | |
|         return $this->billingRequestFlows($data);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Response
 | |
|      *     {
 | |
|      *   "billing_request_flows": {
 | |
|      *     "authorisation_url": "https://pay.gocardless.com/flow/static/billing_request?id=<br_id>",
 | |
|      *     "lock_customer_details": false,
 | |
|      *     "lock_bank_account": false,
 | |
|      *     "auto_fulfil": true,
 | |
|      *     "created_at": "2021-03-30T16:23:10.679Z",
 | |
|      *     "expires_at": "2021-04-06T16:23:10.679Z",
 | |
|      *     "redirect_uri": "https://my-company.com/completed",
 | |
|      *     "links": {
 | |
|      *       "billing_request": "BRQ123"
 | |
|      *     }
 | |
|      *   }
 | |
|      * }
 | |
|      *
 | |
|      *
 | |
|      */
 | |
|     public function billingRequestFlows(array $data)
 | |
|     {
 | |
|         $session_token = \Illuminate\Support\Str::uuid()->toString();
 | |
|         $exit_uri = route('client.payment_methods.index');
 | |
| 
 | |
|         $response = $this->go_cardless->gateway->billingRequests()->create([
 | |
|                         "params" => [
 | |
|                             "mandate_request" => [
 | |
|                             "currency" => auth()->guard('contact')->user()->client->currency()->code,
 | |
|                             "verify" => "when_available"
 | |
|                             ]
 | |
|                         ]
 | |
|                     ]);
 | |
| 
 | |
|         try {
 | |
|             $brf = $this->go_cardless->gateway->billingRequestFlows()->create([
 | |
|                 "params" => [
 | |
|                     "redirect_uri" => route('client.payment_methods.confirm', [
 | |
|                             'method' => GatewayType::DIRECT_DEBIT,
 | |
|                             'session_token' => $session_token,
 | |
|                             'billing_request' => $response->id,
 | |
|                         ]),
 | |
|                     "exit_uri" => $exit_uri,
 | |
|                     "links" => [
 | |
|                     "billing_request" => $response->id
 | |
|                     ],
 | |
|                     "show_redirect_buttons" => true,
 | |
|                     "show_success_redirect_button" => true,
 | |
|                 ]
 | |
|             ]);
 | |
| 
 | |
|             return redirect($brf->authorisation_url);
 | |
| 
 | |
|         } catch (\Exception $exception) {
 | |
|             nlog($exception->getMessage());
 | |
|             return $this->processUnsuccessfulAuthorization($exception);
 | |
|         }
 | |
| 
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /**
 | |
|      * Handle unsuccessful authorization.
 | |
|      *
 | |
|      * @param \Exception $exception
 | |
|      * @throws PaymentFailed
 | |
|      * @return void
 | |
|      */
 | |
|     public function processUnsuccessfulAuthorization(\Exception $exception): void
 | |
|     {
 | |
|         SystemLogger::dispatch(
 | |
|             $exception->getMessage(),
 | |
|             SystemLog::CATEGORY_GATEWAY_RESPONSE,
 | |
|             SystemLog::EVENT_GATEWAY_FAILURE,
 | |
|             SystemLog::TYPE_GOCARDLESS,
 | |
|             $this->go_cardless->client,
 | |
|             $this->go_cardless->client->company,
 | |
|         );
 | |
| 
 | |
|         throw new PaymentFailed($exception->getMessage(), $exception->getCode());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Handle authorization response for Direct Debit.
 | |
|      *
 | |
|      * @param Request $request
 | |
|      * @return RedirectResponse|void
 | |
|      */
 | |
|     public function authorizeResponse(Request $request)
 | |
|     {
 | |
| 
 | |
|         try {
 | |
| 
 | |
|             $billing_request = $this->go_cardless->gateway->billingRequests()->get($request->billing_request);
 | |
| 
 | |
|             $payment_meta = new \stdClass();
 | |
|             $payment_meta->brand = $billing_request->resources->customer_bank_account->bank_name;
 | |
|             $payment_meta->type = $this->resolveScheme($billing_request->mandate_request->scheme);
 | |
|             $payment_meta->state = 'pending';
 | |
|             $payment_meta->last4 = $billing_request->resources->customer_bank_account->account_number_ending;
 | |
| 
 | |
|             $data = [
 | |
|                 'payment_meta' => $payment_meta,
 | |
|                 'token' => $billing_request->mandate_request->links->mandate,
 | |
|                 'payment_method_id' => $this->resolveScheme($billing_request->mandate_request->scheme),
 | |
|             ];
 | |
| 
 | |
|             $payment_method = $this->go_cardless->storeGatewayToken($data, ['gateway_customer_reference' => $billing_request->resources->customer->id]);
 | |
| 
 | |
|             $mandate = $this->go_cardless->gateway->mandates()->get($billing_request->mandate_request->links->mandate);
 | |
| 
 | |
|             nlog($mandate);
 | |
| 
 | |
|             return redirect()->route('client.payment_methods.show', $payment_method->hashed_id);
 | |
| 
 | |
|         } catch (\Exception $exception) {
 | |
|             return $this->processUnsuccessfulAuthorization($exception);
 | |
|         }
 | |
| 
 | |
|         // try {
 | |
|         //     $redirect_flow = $this->go_cardless->gateway->redirectFlows()->complete(
 | |
|         //         $request->redirect_flow_id,
 | |
|         //         ['params' => [
 | |
|         //             'session_token' => $request->session_token,
 | |
|         //         ]],
 | |
|         //     );
 | |
| 
 | |
|         //     $payment_meta = new \stdClass;
 | |
|         //     $payment_meta->brand = ctrans('texts.payment_type_direct_debit');
 | |
|         //     $payment_meta->type = GatewayType::DIRECT_DEBIT;
 | |
|         //     $payment_meta->state = 'authorized';
 | |
| 
 | |
|         //     $data = [
 | |
|         //         'payment_meta' => $payment_meta,
 | |
|         //         'token' => $redirect_flow->links->mandate,
 | |
|         //         'payment_method_id' => $this->resolveScheme($redirect_flow->scheme),
 | |
|         //     ];
 | |
| 
 | |
|         //     $payment_method = $this->go_cardless->storeGatewayToken($data, ['gateway_customer_reference' => $redirect_flow->links->customer]);
 | |
| 
 | |
|         //     return redirect()->route('client.payment_methods.show', $payment_method->hashed_id);
 | |
|         // } catch (\Exception $exception) {
 | |
|         //     return $this->processUnsuccessfulAuthorization($exception);
 | |
|         // }
 | |
|     }
 | |
| 
 | |
|     private function resolveScheme(string $scheme): int
 | |
|     {
 | |
|         match ($scheme) {
 | |
|             'sepa_core' => $type = GatewayType::SEPA,
 | |
|             'ach' => $type = GatewayType::BANK_TRANSFER,
 | |
|             default => $type = GatewayType::DIRECT_DEBIT,
 | |
|         };
 | |
| 
 | |
|         return $type;
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /**
 | |
|      * Payment view for Direct Debit.
 | |
|      *
 | |
|      * @param array $data
 | |
|      * @return \Illuminate\View\View
 | |
|      */
 | |
|     public function paymentView(array $data): View
 | |
|     {
 | |
|         $data['gateway'] = $this->go_cardless;
 | |
|         $data['amount'] = $this->go_cardless->convertToGoCardlessAmount($data['total']['amount_with_fee'], $this->go_cardless->client->currency()->precision);
 | |
|         $data['currency'] = $this->go_cardless->client->getCurrencyCode();
 | |
| 
 | |
|         return render('gateways.gocardless.direct_debit.pay', $data);
 | |
|     }
 | |
| 
 | |
|     public function paymentResponse(PaymentResponseRequest $request)
 | |
|     {
 | |
|         $this->go_cardless->ensureMandateIsReady($request->source);
 | |
| 
 | |
|         $invoice = Invoice::query()->whereIn('id', $this->transformKeys(array_column($this->go_cardless->payment_hash->invoices(), 'invoice_id')))
 | |
|                           ->withTrashed()
 | |
|                           ->first();
 | |
| 
 | |
|         if ($invoice) {
 | |
|             $description = "Invoice {$invoice->number} for {$request->amount} for client {$this->go_cardless->client->present()->name()}";
 | |
|         } else {
 | |
|             $description = "Amount {$request->amount} from client {$this->go_cardless->client->present()->name()}";
 | |
|         }
 | |
| 
 | |
|         $amount = $this->go_cardless->convertToGoCardlessAmount($this->go_cardless->payment_hash?->amount_with_fee(), $this->go_cardless->client->currency()->precision);
 | |
| 
 | |
|         try {
 | |
|             $payment = $this->go_cardless->gateway->payments()->create([
 | |
|                 'params' => [
 | |
|                     // 'amount' => $request->amount,
 | |
|                     'amount' => $amount,
 | |
|                     'currency' => $request->currency,
 | |
|                     'description' => $description,
 | |
|                     'metadata' => [
 | |
|                         'payment_hash' => $this->go_cardless->payment_hash->hash,
 | |
|                     ],
 | |
|                     'links' => [
 | |
|                         'mandate' => $request->source,
 | |
|                     ],
 | |
|                 ],
 | |
|             ]);
 | |
| 
 | |
|             if ($payment->status === 'pending_submission') {
 | |
|                 return $this->processPendingPayment($payment, ['token' => $request->source]);
 | |
|             }
 | |
| 
 | |
|             return $this->processUnsuccessfulPayment($payment);
 | |
|         } catch (\Exception $exception) {
 | |
|             throw new PaymentFailed($exception->getMessage(), $exception->getCode());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Handle pending payments for Direct Debit.
 | |
|      *
 | |
|      * @param ResourcesPayment $payment
 | |
|      * @param array $data
 | |
|      * @return RedirectResponse
 | |
|      */
 | |
|     public function processPendingPayment(\GoCardlessPro\Resources\Payment $payment, array $data = [])
 | |
|     {
 | |
|         $data = [
 | |
|             'payment_type' => PaymentType::DIRECT_DEBIT,
 | |
|             'amount' => $this->go_cardless->payment_hash->data->amount_with_fee,
 | |
|             'transaction_reference' => $payment->id,
 | |
|             'gateway_type_id' => GatewayType::DIRECT_DEBIT,
 | |
|         ];
 | |
| 
 | |
|         $payment = $this->go_cardless->createPayment($data, Payment::STATUS_PENDING);
 | |
| 
 | |
|         SystemLogger::dispatch(
 | |
|             ['response' => $payment, 'data' => $data],
 | |
|             SystemLog::CATEGORY_GATEWAY_RESPONSE,
 | |
|             SystemLog::EVENT_GATEWAY_SUCCESS,
 | |
|             SystemLog::TYPE_GOCARDLESS,
 | |
|             $this->go_cardless->client,
 | |
|             $this->go_cardless->client->company,
 | |
|         );
 | |
| 
 | |
|         return redirect()->route('client.payments.show', ['payment' => $this->go_cardless->encodePrimaryKey($payment->id)]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Process unsuccessful payments for Direct Debit.
 | |
|      *
 | |
|      * @param ResourcesPayment $payment
 | |
|      * @return never
 | |
|      */
 | |
|     public function processUnsuccessfulPayment(\GoCardlessPro\Resources\Payment $payment)
 | |
|     {
 | |
|         PaymentFailureMailer::dispatch($this->go_cardless->client, $payment->status, $this->go_cardless->client->company, $this->go_cardless->payment_hash->data->amount_with_fee);
 | |
| 
 | |
|         PaymentFailureMailer::dispatch(
 | |
|             $this->go_cardless->client,
 | |
|             $payment,
 | |
|             $this->go_cardless->client->company,
 | |
|             $payment->amount
 | |
|         );
 | |
| 
 | |
|         $message = [
 | |
|             'server_response' => $payment,
 | |
|             'data' => $this->go_cardless->payment_hash->data,
 | |
|         ];
 | |
| 
 | |
|         SystemLogger::dispatch(
 | |
|             $message,
 | |
|             SystemLog::CATEGORY_GATEWAY_RESPONSE,
 | |
|             SystemLog::EVENT_GATEWAY_FAILURE,
 | |
|             SystemLog::TYPE_GOCARDLESS,
 | |
|             $this->go_cardless->client,
 | |
|             $this->go_cardless->client->company,
 | |
|         );
 | |
| 
 | |
|         throw new PaymentFailed('Failed to process the payment.', 500);
 | |
|     }
 | |
| }
 |