1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-11-18 00:53:10 +01:00
invoiceninja/app/Http/Controllers/ClientPortal/PaymentController.php

360 lines
13 KiB
PHP
Raw Normal View History

2019-08-16 07:20:28 +02:00
<?php
2019-08-16 07:20:28 +02:00
/**
* Invoice Ninja (https://invoiceninja.com).
2019-08-16 07:20:28 +02:00
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
2019-08-16 07:20:28 +02:00
*
* @license https://opensource.org/licenses/AAL
*/
namespace App\Http\Controllers\ClientPortal;
use App\Exceptions\PaymentFailed;
2020-10-13 14:28:30 +02:00
use App\Factory\PaymentFactory;
2019-08-16 07:20:28 +02:00
use App\Http\Controllers\Controller;
use App\Http\Requests\ClientPortal\Payments\PaymentResponseRequest;
use App\Jobs\Invoice\InjectSignature;
use App\Jobs\Util\SystemLogger;
2019-09-13 00:33:48 +02:00
use App\Models\CompanyGateway;
2019-09-25 04:07:33 +02:00
use App\Models\Invoice;
2019-08-16 07:20:28 +02:00
use App\Models\Payment;
2020-08-26 02:47:50 +02:00
use App\Models\PaymentHash;
use App\Models\SystemLog;
use App\Services\Subscription\SubscriptionService;
2019-09-25 04:07:33 +02:00
use App\Utils\Number;
use App\Utils\Traits\MakesDates;
2019-08-16 07:20:28 +02:00
use App\Utils\Traits\MakesHash;
2020-10-28 11:10:49 +01:00
use Illuminate\Contracts\View\Factory;
use Illuminate\Http\RedirectResponse;
2019-08-16 07:20:28 +02:00
use Illuminate\Http\Request;
2021-03-17 16:12:25 +01:00
use Illuminate\Support\Facades\Cache;
2020-08-26 02:47:50 +02:00
use Illuminate\Support\Str;
2020-10-28 11:10:49 +01:00
use Illuminate\View\View;
2019-08-16 07:20:28 +02:00
/**
* Class PaymentController.
2019-08-16 07:20:28 +02:00
*/
class PaymentController extends Controller
{
use MakesHash;
2019-09-25 04:07:33 +02:00
use MakesDates;
2019-08-16 07:20:28 +02:00
/**
* Show the list of payments.
2019-08-16 07:20:28 +02:00
*
2020-10-28 11:10:49 +01:00
* @return Factory|View
2019-08-16 07:20:28 +02:00
*/
public function index()
2019-08-16 07:20:28 +02:00
{
return $this->render('payments.index');
2019-08-16 07:20:28 +02:00
}
/**
* Display the specified resource.
*
2020-03-23 18:10:42 +01:00
* @param Request $request
* @param Payment $payment
2020-10-28 11:10:49 +01:00
* @return Factory|View
2019-08-16 07:20:28 +02:00
*/
public function show(Request $request, Payment $payment)
2019-08-16 07:20:28 +02:00
{
$payment->load('invoices');
2019-08-16 07:20:28 +02:00
2020-03-23 18:10:42 +01:00
return $this->render('payments.show', [
'payment' => $payment,
]);
2019-08-16 07:20:28 +02:00
}
2019-09-12 08:10:21 +02:00
/**
* Presents the payment screen for a given
* gateway and payment method.
* The request will also contain the amount
2019-09-12 13:46:09 +02:00
* and invoice ids for reference.
*
2020-10-28 11:10:49 +01:00
* @param Request $request
* @return RedirectResponse|mixed
2019-09-12 08:10:21 +02:00
*/
2020-10-15 02:37:16 +02:00
public function process(Request $request)
{
2020-10-15 05:35:35 +02:00
$is_credit_payment = false;
$tokens = [];
2020-10-15 05:35:35 +02:00
2020-11-25 15:19:52 +01:00
if ($request->input('company_gateway_id') == CompanyGateway::GATEWAY_CREDIT) {
2020-10-15 05:35:35 +02:00
$is_credit_payment = true;
2020-11-25 15:19:52 +01:00
}
2020-10-15 02:37:16 +02:00
2020-10-15 05:35:35 +02:00
$gateway = CompanyGateway::find($request->input('company_gateway_id'));
2020-09-24 11:29:47 +02:00
/**
* find invoices
*
* ['invoice_id' => xxx, 'amount' => 22.00]
*/
2020-10-15 05:35:35 +02:00
$payable_invoices = collect($request->payable_invoices);
$invoices = Invoice::whereIn('id', $this->transformKeys($payable_invoices->pluck('invoice_id')->toArray()))->get();
2020-09-24 11:29:47 +02:00
/* pop non payable invoice from the $payable_invoices array */
2020-11-25 15:19:52 +01:00
$payable_invoices = $payable_invoices->filter(function ($payable_invoice) use ($invoices) {
return $invoices->where('hashed_id', $payable_invoice['invoice_id'])->first()->isPayable();
});
2020-08-25 15:18:17 +02:00
/*return early if no invoices*/
if ($payable_invoices->count() == 0) {
return redirect()
->route('client.invoices.index')
->with(['message' => 'No payable invoices selected.']);
}
2020-03-23 18:10:42 +01:00
2020-09-24 11:29:47 +02:00
$settings = auth()->user()->client->getMergedSettings();
2020-08-26 02:53:11 +02:00
// nlog($settings);
2021-01-06 06:54:04 +01:00
2021-01-06 06:14:20 +01:00
/* This loop checks for under / over payments and returns the user if a check fails */
2021-01-25 16:46:40 +01:00
foreach ($payable_invoices as $payable_invoice) {
2021-01-06 06:14:20 +01:00
/*Match the payable invoice to the Model Invoice*/
2020-08-26 02:47:50 +02:00
$invoice = $invoices->first(function ($inv) use ($payable_invoice) {
return $payable_invoice['invoice_id'] == $inv->hashed_id;
});
2021-01-06 06:14:20 +01:00
/*
* Check if company supports over & under payments.
* Determine the payable amount and the max payable. ie either partial or invoice balance
*/
2020-09-24 11:29:47 +02:00
$payable_amount = Number::roundValue(Number::parseFloat($payable_invoice['amount']), auth()->user()->client->currency()->precision);
2021-01-06 06:14:20 +01:00
$invoice_balance = Number::roundValue(($invoice->partial > 0 ? $invoice->partial : $invoice->balance), auth()->user()->client->currency()->precision);
/*If we don't allow under/over payments force the payable amount - prevents inspect element adjustments in JS*/
2020-09-24 11:29:47 +02:00
if ($settings->client_portal_allow_under_payment == false && $settings->client_portal_allow_over_payment == false) {
2020-09-29 14:16:01 +02:00
$payable_invoice['amount'] = Number::roundValue(($invoice->partial > 0 ? $invoice->partial : $invoice->balance), auth()->user()->client->currency()->precision);
}
2021-01-06 06:14:20 +01:00
2021-02-09 12:48:22 +01:00
if (!$settings->client_portal_allow_under_payment && $payable_amount < $invoice_balance) {
return redirect()
->route('client.invoices.index')
->with('message', ctrans('texts.minimum_required_payment', ['amount' => $invoice_balance]));
}
2020-09-24 11:29:47 +02:00
if ($settings->client_portal_allow_under_payment) {
2021-02-09 12:48:22 +01:00
if ($invoice_balance < $settings->client_portal_under_payment_minimum && $payable_amount < $invoice_balance) {
2020-09-24 11:29:47 +02:00
return redirect()
->route('client.invoices.index')
2021-02-09 12:48:22 +01:00
->with('message', ctrans('texts.minimum_required_payment', ['amount' => $invoice_balance]));
}
if ($invoice_balance < $settings->client_portal_under_payment_minimum) {
// Skip the under payment rule.
2020-09-24 11:29:47 +02:00
}
2021-02-09 12:48:22 +01:00
if ($invoice_balance >= $settings->client_portal_under_payment_minimum && $payable_amount < $settings->client_portal_under_payment_minimum) {
2020-09-24 11:29:47 +02:00
return redirect()
->route('client.invoices.index')
2021-02-09 12:48:22 +01:00
->with('message', ctrans('texts.minimum_required_payment', ['amount' => $settings->client_portal_under_payment_minimum]));
2020-09-24 11:29:47 +02:00
}
}
2021-01-06 06:14:20 +01:00
/* If we don't allow over payments and the amount exceeds the balance */
2020-09-24 11:29:47 +02:00
2021-02-09 12:48:22 +01:00
if (!$settings->client_portal_allow_over_payment && $payable_amount > $invoice_balance) {
return redirect()
->route('client.invoices.index')
->with('message', ctrans('texts.over_payments_disabled'));
}
2021-01-06 06:14:20 +01:00
}
/*Iterate through invoices and add gateway fees and other payment metadata*/
2021-01-07 08:21:49 +01:00
//$payable_invoices = $payable_invoices->map(function ($payable_invoice) use ($invoices, $settings) {
$payable_invoice_collection = collect();
2021-01-25 16:46:40 +01:00
foreach ($payable_invoices as $payable_invoice) {
2021-01-14 05:31:45 +01:00
// nlog($payable_invoice);
2021-01-07 08:21:49 +01:00
2021-01-06 06:14:20 +01:00
$payable_invoice['amount'] = Number::parseFloat($payable_invoice['amount']);
$invoice = $invoices->first(function ($inv) use ($payable_invoice) {
return $payable_invoice['invoice_id'] == $inv->hashed_id;
});
$payable_amount = Number::roundValue(Number::parseFloat($payable_invoice['amount']), auth()->user()->client->currency()->precision);
$invoice_balance = Number::roundValue($invoice->balance, auth()->user()->client->currency()->precision);
2020-09-24 11:29:47 +02:00
$payable_invoice['due_date'] = $this->formatDate($invoice->due_date, $invoice->client->date_format());
$payable_invoice['invoice_number'] = $invoice->number;
2020-08-26 02:47:50 +02:00
if (isset($invoice->po_number)) {
2020-08-26 02:47:50 +02:00
$additional_info = $invoice->po_number;
} elseif (isset($invoice->public_notes)) {
2020-08-26 02:47:50 +02:00
$additional_info = $invoice->public_notes;
} else {
2020-08-26 02:47:50 +02:00
$additional_info = $invoice->date;
}
2020-08-26 02:47:50 +02:00
$payable_invoice['additional_info'] = $additional_info;
2021-01-07 08:21:49 +01:00
$payable_invoice_collection->push($payable_invoice);
}
//});
if (request()->has('signature') && !is_null(request()->signature) && !empty(request()->signature)) {
$invoices->each(function ($invoice) use ($request) {
2020-10-15 05:35:35 +02:00
InjectSignature::dispatch($invoice, $request->signature);
2020-06-01 14:29:41 +02:00
});
}
2021-01-07 08:21:49 +01:00
$payable_invoices = $payable_invoice_collection;
2020-10-15 05:35:35 +02:00
$payment_method_id = $request->input('payment_method_id');
$invoice_totals = $payable_invoices->sum('amount');
$first_invoice = $invoices->first();
2021-01-11 22:42:30 +01:00
$credit_totals = $first_invoice->client->getSetting('use_credits_payment') == 'always' ? $first_invoice->client->service()->getCreditBalance() : 0;
$starting_invoice_amount = $first_invoice->amount;
2020-11-25 15:19:52 +01:00
if ($gateway) {
2020-10-15 05:35:35 +02:00
$first_invoice->service()->addGatewayFee($gateway, $payment_method_id, $invoice_totals)->save();
2020-11-25 15:19:52 +01:00
}
/**
2020-10-28 11:10:49 +01:00
* Gateway fee is calculated
* by adding it as a line item, and then subtract
2020-10-15 02:37:16 +02:00
* the starting and finishing amounts of the invoice.
*/
$fee_totals = $first_invoice->amount - $starting_invoice_amount;
2020-11-25 15:19:52 +01:00
if ($gateway) {
$tokens = auth()->user()->client->gateway_tokens()
->whereCompanyGatewayId($gateway->id)
->whereGatewayTypeId($payment_method_id)
->get();
2020-11-25 15:19:52 +01:00
}
2020-10-15 05:35:35 +02:00
2021-03-17 16:12:25 +01:00
$hash_data = ['invoices' => $payable_invoices->toArray(), 'credits' => $credit_totals];
if ($request->query('hash')) {
$hash_data['billing_context'] = Cache::get($request->query('hash'));
}
2020-08-26 02:47:50 +02:00
$payment_hash = new PaymentHash;
2021-04-22 13:22:55 +02:00
$payment_hash->hash = Str::random(32);
2021-03-17 16:12:25 +01:00
$payment_hash->data = $hash_data;
2020-08-30 14:00:19 +02:00
$payment_hash->fee_total = $fee_totals;
2020-08-31 06:27:47 +02:00
$payment_hash->fee_invoice_id = $first_invoice->id;
2021-03-17 16:12:25 +01:00
2020-08-26 02:47:50 +02:00
$payment_hash->save();
$totals = [
'credit_totals' => $credit_totals,
2020-08-26 03:14:15 +02:00
'invoice_totals' => $invoice_totals,
'fee_total' => $fee_totals,
'amount_with_fee' => max(0, (($invoice_totals + $fee_totals) - $credit_totals)),
2020-08-26 02:47:50 +02:00
];
2020-03-23 18:10:42 +01:00
2019-09-12 08:10:21 +02:00
$data = [
2020-08-26 02:47:50 +02:00
'payment_hash' => $payment_hash->hash,
'total' => $totals,
'invoices' => $payable_invoices,
'tokens' => $tokens,
2019-09-25 04:07:33 +02:00
'payment_method_id' => $payment_method_id,
2020-08-30 14:00:19 +02:00
'amount_with_fee' => $invoice_totals + $fee_totals,
2019-09-12 08:10:21 +02:00
];
2020-03-23 18:10:42 +01:00
2020-11-25 15:19:52 +01:00
if ($is_credit_payment) {
2020-10-15 05:35:35 +02:00
return $this->processCreditPayment($request, $data);
}
2020-10-15 05:35:35 +02:00
try {
return $gateway
->driver(auth()->user()->client)
->setPaymentMethod($payment_method_id)
->setPaymentHash($payment_hash)
->checkRequirements()
->processPaymentView($data);
2021-01-25 16:46:40 +01:00
} catch (\Exception $e) {
SystemLogger::dispatch(
$e->getMessage(),
SystemLog::CATEGORY_GATEWAY_RESPONSE,
SystemLog::EVENT_GATEWAY_ERROR,
SystemLog::TYPE_FAILURE,
auth('contact')->user()->client
);
throw new PaymentFailed($e->getMessage());
}
2019-09-12 08:10:21 +02:00
}
public function response(PaymentResponseRequest $request)
{
2020-12-30 12:02:04 +01:00
$gateway = CompanyGateway::findOrFail($request->input('company_gateway_id'));
2020-12-09 15:17:48 +01:00
$payment_hash = PaymentHash::whereRaw('BINARY `hash`= ?', [$request->payment_hash])->first();
2021-01-25 16:46:40 +01:00
try {
return $gateway
->driver(auth()->user()->client)
->setPaymentMethod($request->input('payment_method_id'))
->setPaymentHash($payment_hash)
->checkRequirements()
->processPaymentResponse($request);
2021-01-25 16:46:40 +01:00
} catch (\Exception $e) {
SystemLogger::dispatch(
$e->getMessage(),
SystemLog::CATEGORY_GATEWAY_RESPONSE,
SystemLog::EVENT_GATEWAY_FAILURE,
SystemLog::TYPE_FAILURE,
auth('contact')->user()->client
2021-01-25 16:46:40 +01:00
);
throw new PaymentFailed($e->getMessage());
2021-01-25 16:46:40 +01:00
}
}
2020-10-14 01:53:20 +02:00
/**
* Pay for invoice/s using credits only.
2020-10-28 11:10:49 +01:00
*
2021-01-25 16:46:40 +01:00
* @param Request $request The request object
2020-10-14 01:53:20 +02:00
* @return Response The response view
*/
public function credit_response(Request $request)
{
2020-10-14 01:53:20 +02:00
$payment_hash = PaymentHash::whereRaw('BINARY `hash`= ?', [$request->input('payment_hash')])->first();
2020-10-13 14:28:30 +02:00
2020-10-14 01:53:20 +02:00
/* Hydrate the $payment */
2020-11-25 15:19:52 +01:00
if ($payment_hash->payment()->exists()) {
2020-10-13 14:28:30 +02:00
$payment = $payment_hash->payment;
2020-11-25 15:19:52 +01:00
} else {
2020-10-14 01:53:20 +02:00
$payment = PaymentFactory::create($payment_hash->fee_invoice->company_id, $payment_hash->fee_invoice->user_id);
$payment->client_id = $payment_hash->fee_invoice->client_id;
$payment->save();
2020-10-13 14:28:30 +02:00
$payment_hash->payment_id = $payment->id;
$payment_hash->save();
}
2021-01-06 06:54:04 +01:00
$payment = $payment->service()->applyCredits($payment_hash)->save();
2020-10-13 14:28:30 +02:00
if (property_exists($payment_hash->data, 'billing_context')) {
$billing_subscription = \App\Models\Subscription::find($payment_hash->data->billing_context->subscription_id);
return (new SubscriptionService($billing_subscription))->completePurchase($payment_hash);
}
2020-10-14 01:53:20 +02:00
return redirect()->route('client.payments.show', ['payment' => $this->encodePrimaryKey($payment->id)]);
}
2020-10-15 02:37:16 +02:00
2020-10-15 05:35:35 +02:00
public function processCreditPayment(Request $request, array $data)
2020-10-15 02:37:16 +02:00
{
2020-10-15 05:35:35 +02:00
return render('gateways.credit.index', $data);
2020-10-15 02:37:16 +02:00
}
2019-08-16 07:20:28 +02:00
}