1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-09-22 01:11:34 +02:00
invoiceninja/app/Services/Payment/DeletePayment.php

183 lines
5.7 KiB
PHP
Raw Normal View History

2020-06-28 00:24:08 +02:00
<?php
2020-07-01 03:06:40 +02:00
/**
* Invoice Ninja (https://invoiceninja.com).
2020-07-01 03:06:40 +02:00
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
2023-01-28 23:21:40 +01:00
* @copyright Copyright (c) 2023. Invoice Ninja LLC (https://invoiceninja.com)
2020-07-01 03:06:40 +02:00
*
2021-06-16 08:58:16 +02:00
* @license https://www.elastic.co/licensing/elastic-license
2020-07-01 03:06:40 +02:00
*/
2020-06-28 00:24:08 +02:00
namespace App\Services\Payment;
use App\Models\Credit;
use App\Models\Invoice;
use App\Models\Payment;
use Illuminate\Contracts\Container\BindingResolutionException;
2020-06-28 00:24:08 +02:00
class DeletePayment
{
private float $_paid_to_date_deleted = 0;
2020-06-28 00:24:08 +02:00
/**
2023-02-16 02:36:09 +01:00
* @param mixed $payment
* @return void
*/
2023-02-16 02:36:09 +01:00
public function __construct(public Payment $payment, private bool $update_client_paid_to_date)
{
}
2020-06-28 00:24:08 +02:00
/**
2023-02-16 02:36:09 +01:00
* @return mixed
* @throws BindingResolutionException
*/
2020-06-28 00:24:08 +02:00
public function run()
{
\DB::connection(config('database.default'))->transaction(function () {
2022-10-05 02:21:55 +02:00
$this->payment = Payment::withTrashed()->where('id', $this->payment->id)->lockForUpdate()->first();
2023-01-15 04:59:42 +01:00
if ($this->payment && !$this->payment->is_deleted) {
$this->setStatus(Payment::STATUS_CANCELLED) //sets status of payment
->updateCreditables() //return the credits first
->adjustInvoices()
->deletePaymentables()
->cleanupPayment()
->save();
}
}, 2);
2022-10-05 02:21:55 +02:00
return $this->payment;
2020-06-28 00:24:08 +02:00
}
/** @return $this */
2020-11-23 13:55:04 +01:00
private function cleanupPayment()
{
$this->payment->is_deleted = true;
$this->payment->delete();
2020-11-23 13:55:04 +01:00
return $this;
}
/** @return $this */
2020-11-15 22:23:20 +01:00
private function deletePaymentables()
{
$this->payment->paymentables()->update(['deleted_at' => now()]);
return $this;
}
/** @return $this */
2020-06-28 05:05:58 +02:00
private function adjustInvoices()
{
$this->_paid_to_date_deleted = 0;
if ($this->payment->invoices()->exists()) {
$this->payment->invoices()->each(function ($paymentable_invoice) {
$net_deletable = $paymentable_invoice->pivot->amount - $paymentable_invoice->pivot->refunded;
$this->_paid_to_date_deleted += $net_deletable;
$paymentable_invoice = $paymentable_invoice->fresh();
2022-03-09 03:49:31 +01:00
2021-09-04 07:51:31 +02:00
nlog("net deletable amount - refunded = {$net_deletable}");
if (! $paymentable_invoice->is_deleted) {
2022-03-14 10:52:38 +01:00
$paymentable_invoice->restore();
$paymentable_invoice->service()
->updateBalance($net_deletable)
->updatePaidToDate($net_deletable * -1)
->save();
$paymentable_invoice->ledger()
->updateInvoiceBalance($net_deletable, "Adjusting invoice {$paymentable_invoice->number} due to deletion of Payment {$this->payment->number}")
->save();
$this->payment
->client
->service()
->updateBalanceAndPaidToDate($net_deletable, $net_deletable*-1)
->save();
if ($paymentable_invoice->balance == $paymentable_invoice->amount) {
$paymentable_invoice->service()->setStatus(Invoice::STATUS_SENT)->save();
} else {
$paymentable_invoice->service()->setStatus(Invoice::STATUS_PARTIAL)->save();
}
} else {
$paymentable_invoice->restore();
2021-09-06 07:08:41 +02:00
$paymentable_invoice->service()
->updatePaidToDate($net_deletable * -1)
->save();
}
2020-06-28 05:05:58 +02:00
});
}
//sometimes the payment is NOT created properly, this catches the payment and prevents the paid to date reducing inappropriately.
2023-02-16 02:36:09 +01:00
if ($this->update_client_paid_to_date) {
$this->payment
->client
->service()
->updatePaidToDate(min(0, ($this->payment->amount - $this->payment->refunded - $this->_paid_to_date_deleted) * -1))
->save();
}
2020-06-28 05:05:58 +02:00
return $this;
}
/** @return $this */
2020-06-28 05:05:58 +02:00
private function updateCreditables()
{
if ($this->payment->credits()->exists()) {
2023-02-16 02:36:09 +01:00
$this->payment->credits()->where('is_deleted', 0)->each(function ($paymentable_credit) {
2021-11-22 11:09:28 +01:00
$multiplier = 1;
if ($paymentable_credit->pivot->amount < 0) {
2022-03-16 03:06:25 +01:00
$multiplier = -1;
}
2021-11-22 11:09:28 +01:00
$paymentable_credit->service()
->updateBalance($paymentable_credit->pivot->amount * $multiplier * -1)
->updatePaidToDate($paymentable_credit->pivot->amount * $multiplier)
->setStatus(Credit::STATUS_SENT)
->save();
2022-03-01 11:25:18 +01:00
2022-03-16 03:06:25 +01:00
$client = $this->payment->client->fresh();
$client
->service()
->updatePaidToDate(($paymentable_credit->pivot->amount) * -1)
->adjustCreditBalance($paymentable_credit->pivot->amount)
2022-03-16 03:06:25 +01:00
->save();
2020-06-28 05:05:58 +02:00
});
}
return $this;
}
/**
2023-02-16 02:36:09 +01:00
* @param mixed $status
* @return $this
*/
2020-06-28 05:05:58 +02:00
private function setStatus($status)
{
2020-07-06 13:22:36 +02:00
$this->payment->status_id = Payment::STATUS_CANCELLED;
2020-06-28 05:05:58 +02:00
return $this;
}
2020-06-28 00:24:08 +02:00
/**
* Saves the payment.
*
2020-06-28 00:24:08 +02:00
* @return Payment $payment
*/
private function save()
{
$this->payment->save();
return $this->payment;
}
}