2020-02-03 11:33:07 +01:00
|
|
|
<?php
|
|
|
|
/**
|
2021-02-18 00:51:56 +01:00
|
|
|
* Invoice Ninja (https://invoiceninja.com).
|
2020-02-03 11:33:07 +01:00
|
|
|
*
|
2021-02-18 00:51:56 +01:00
|
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
2020-02-03 11:33:07 +01:00
|
|
|
*
|
2023-01-28 23:21:40 +01:00
|
|
|
* @copyright Copyright (c) 2023. Invoice Ninja LLC (https://invoiceninja.com)
|
2020-02-03 11:33:07 +01:00
|
|
|
*
|
2021-06-16 08:58:16 +02:00
|
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
2020-02-03 11:33:07 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
namespace App\Services\Client;
|
|
|
|
|
2024-01-28 05:05:30 +01:00
|
|
|
use App\Utils\Number;
|
2020-02-03 11:33:07 +01:00
|
|
|
use App\Models\Client;
|
2022-09-05 09:18:08 +02:00
|
|
|
use App\Models\Credit;
|
2023-10-27 08:13:23 +02:00
|
|
|
use App\Models\Invoice;
|
2023-03-10 04:01:07 +01:00
|
|
|
use App\Models\Payment;
|
2023-03-07 09:52:37 +01:00
|
|
|
use App\Services\Email\Email;
|
2023-11-26 08:41:42 +01:00
|
|
|
use App\Utils\Traits\MakesDates;
|
|
|
|
use Illuminate\Support\Facades\DB;
|
2024-01-28 05:05:30 +01:00
|
|
|
use App\Services\Email\EmailObject;
|
|
|
|
use App\Utils\Traits\GeneratesCounter;
|
|
|
|
use Illuminate\Mail\Mailables\Address;
|
|
|
|
use Illuminate\Database\QueryException;
|
2020-02-03 11:33:07 +01:00
|
|
|
|
|
|
|
class ClientService
|
|
|
|
{
|
2024-01-28 05:05:30 +01:00
|
|
|
use MakesDates, GeneratesCounter;
|
2020-02-03 11:33:07 +01:00
|
|
|
|
2023-01-17 01:00:12 +01:00
|
|
|
private string $client_start_date;
|
|
|
|
|
|
|
|
private string $client_end_date;
|
|
|
|
|
2024-01-28 05:05:30 +01:00
|
|
|
private bool $completed = true;
|
|
|
|
|
2023-02-16 02:36:09 +01:00
|
|
|
public function __construct(private Client $client)
|
|
|
|
{
|
|
|
|
}
|
2020-02-03 11:33:07 +01:00
|
|
|
|
2023-11-08 09:03:56 +01:00
|
|
|
public function calculateBalance(?Invoice $invoice = null)
|
2023-11-07 02:10:16 +01:00
|
|
|
{
|
2023-11-12 22:18:30 +01:00
|
|
|
$balance = Invoice::withTrashed()
|
|
|
|
->where('client_id', $this->client->id)
|
2023-11-07 02:10:16 +01:00
|
|
|
->whereIn('status_id', [Invoice::STATUS_SENT, Invoice::STATUS_PARTIAL])
|
|
|
|
->where('is_deleted', false)
|
|
|
|
->sum('balance');
|
|
|
|
|
2023-11-08 09:03:56 +01:00
|
|
|
$pre_client_balance = $this->client->balance;
|
|
|
|
|
2023-11-07 02:10:16 +01:00
|
|
|
try {
|
|
|
|
DB::connection(config('database.default'))->transaction(function () use ($balance) {
|
|
|
|
$this->client = Client::withTrashed()->where('id', $this->client->id)->lockForUpdate()->first();
|
|
|
|
$this->client->balance = $balance;
|
|
|
|
$this->client->saveQuietly();
|
|
|
|
}, 2);
|
|
|
|
} catch (\Throwable $throwable) {
|
|
|
|
nlog("DB ERROR " . $throwable->getMessage());
|
|
|
|
}
|
2024-01-14 05:05:00 +01:00
|
|
|
|
2023-11-08 09:03:56 +01:00
|
|
|
if($invoice && floatval($this->client->balance) != floatval($pre_client_balance)) {
|
|
|
|
$diff = $this->client->balance - $pre_client_balance;
|
|
|
|
$invoice->ledger()->insertInvoiceBalance($diff, $this->client->balance, "Update Adjustment Invoice # {$invoice->number} => {$diff}");
|
|
|
|
}
|
|
|
|
|
2023-11-07 02:10:16 +01:00
|
|
|
return $this;
|
|
|
|
}
|
2024-01-14 05:05:00 +01:00
|
|
|
|
2023-11-08 09:03:56 +01:00
|
|
|
/**
|
|
|
|
* Seeing too many race conditions under heavy load here.
|
2023-11-26 08:41:42 +01:00
|
|
|
*
|
2023-11-08 09:03:56 +01:00
|
|
|
* @param float $amount
|
2023-11-08 09:53:38 +01:00
|
|
|
* @return ClientService
|
2023-11-08 09:03:56 +01:00
|
|
|
*/
|
2020-02-03 11:33:07 +01:00
|
|
|
public function updateBalance(float $amount)
|
|
|
|
{
|
2022-11-24 10:33:52 +01:00
|
|
|
try {
|
2023-02-16 02:36:09 +01:00
|
|
|
DB::connection(config('database.default'))->transaction(function () use ($amount) {
|
2022-11-24 10:33:52 +01:00
|
|
|
$this->client = Client::withTrashed()->where('id', $this->client->id)->lockForUpdate()->first();
|
|
|
|
$this->client->balance += $amount;
|
2023-02-01 03:46:39 +01:00
|
|
|
$this->client->saveQuietly();
|
2023-01-18 06:52:32 +01:00
|
|
|
}, 2);
|
2023-02-16 02:36:09 +01:00
|
|
|
} catch (\Throwable $throwable) {
|
2023-10-13 12:14:37 +02:00
|
|
|
|
|
|
|
if (DB::connection(config('database.default'))->transactionLevel() > 0) {
|
|
|
|
DB::connection(config('database.default'))->rollBack();
|
|
|
|
}
|
|
|
|
|
2023-10-26 04:57:44 +02:00
|
|
|
} catch(\Exception $exception) {
|
2023-10-13 12:14:37 +02:00
|
|
|
|
|
|
|
if (DB::connection(config('database.default'))->transactionLevel() > 0) {
|
|
|
|
DB::connection(config('database.default'))->rollBack();
|
|
|
|
}
|
2022-11-24 10:33:52 +01:00
|
|
|
}
|
2022-09-05 03:51:47 +02:00
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function updateBalanceAndPaidToDate(float $balance, float $paid_to_date)
|
|
|
|
{
|
2022-11-24 10:33:52 +01:00
|
|
|
try {
|
2023-02-16 02:36:09 +01:00
|
|
|
DB::connection(config('database.default'))->transaction(function () use ($balance, $paid_to_date) {
|
2022-11-24 10:33:52 +01:00
|
|
|
$this->client = Client::withTrashed()->where('id', $this->client->id)->lockForUpdate()->first();
|
|
|
|
$this->client->balance += $balance;
|
|
|
|
$this->client->paid_to_date += $paid_to_date;
|
2023-02-01 03:46:39 +01:00
|
|
|
$this->client->saveQuietly();
|
2023-01-18 06:52:32 +01:00
|
|
|
}, 2);
|
2023-02-16 02:36:09 +01:00
|
|
|
} catch (\Throwable $throwable) {
|
2022-11-24 10:33:52 +01:00
|
|
|
nlog("DB ERROR " . $throwable->getMessage());
|
2023-10-13 12:14:37 +02:00
|
|
|
|
|
|
|
if (DB::connection(config('database.default'))->transactionLevel() > 0) {
|
|
|
|
DB::connection(config('database.default'))->rollBack();
|
|
|
|
}
|
|
|
|
|
2023-10-26 04:57:44 +02:00
|
|
|
} catch(\Exception $exception) {
|
2023-10-13 12:14:37 +02:00
|
|
|
nlog("DB ERROR " . $exception->getMessage());
|
|
|
|
|
|
|
|
if (DB::connection(config('database.default'))->transactionLevel() > 0) {
|
|
|
|
DB::connection(config('database.default'))->rollBack();
|
|
|
|
}
|
2022-11-24 10:33:52 +01:00
|
|
|
}
|
2024-01-14 05:05:00 +01:00
|
|
|
|
2020-02-03 11:33:07 +01:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function updatePaidToDate(float $amount)
|
|
|
|
{
|
2023-10-13 12:14:37 +02:00
|
|
|
try {
|
|
|
|
DB::connection(config('database.default'))->transaction(function () use ($amount) {
|
|
|
|
$this->client = Client::withTrashed()->where('id', $this->client->id)->lockForUpdate()->first();
|
|
|
|
$this->client->paid_to_date += $amount;
|
|
|
|
$this->client->saveQuietly();
|
|
|
|
}, 2);
|
2023-10-26 04:57:44 +02:00
|
|
|
} catch (\Throwable $throwable) {
|
2023-10-13 12:14:37 +02:00
|
|
|
nlog("DB ERROR " . $throwable->getMessage());
|
|
|
|
|
|
|
|
if (DB::connection(config('database.default'))->transactionLevel() > 0) {
|
|
|
|
DB::connection(config('database.default'))->rollBack();
|
|
|
|
}
|
|
|
|
|
2023-10-26 04:57:44 +02:00
|
|
|
} catch(\Exception $exception) {
|
2023-10-13 12:14:37 +02:00
|
|
|
nlog("DB ERROR " . $exception->getMessage());
|
|
|
|
|
|
|
|
if (DB::connection(config('database.default'))->transactionLevel() > 0) {
|
|
|
|
DB::connection(config('database.default'))->rollBack();
|
|
|
|
}
|
|
|
|
}
|
2020-02-03 11:33:07 +01:00
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2024-01-28 05:05:30 +01:00
|
|
|
public function applyNumber(): self
|
|
|
|
{
|
|
|
|
$x = 1;
|
|
|
|
|
|
|
|
if(isset($this->client->number))
|
|
|
|
return $this;
|
|
|
|
|
|
|
|
do {
|
|
|
|
try {
|
|
|
|
$this->client->number = $this->getNextClientNumber($this->client);
|
|
|
|
$this->client->saveQuietly();
|
|
|
|
|
|
|
|
$this->completed = false;
|
|
|
|
} catch (QueryException $e) {
|
|
|
|
$x++;
|
|
|
|
|
|
|
|
if ($x > 10) {
|
|
|
|
$this->completed = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while ($this->completed);
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2023-03-10 04:01:07 +01:00
|
|
|
public function updatePaymentBalance()
|
|
|
|
{
|
2023-08-06 09:35:19 +02:00
|
|
|
$amount = Payment::query()->where('client_id', $this->client->id)
|
2023-03-10 04:01:07 +01:00
|
|
|
->where('is_deleted', 0)
|
|
|
|
->whereIn('status_id', [Payment::STATUS_COMPLETED, Payment::STATUS_PENDING, Payment::STATUS_PARTIALLY_REFUNDED, Payment::STATUS_REFUNDED])
|
2023-08-27 01:49:41 +02:00
|
|
|
->selectRaw('SUM(payments.amount - payments.applied) as amount')->first()->amount ?? 0;
|
2023-03-10 04:01:07 +01:00
|
|
|
|
|
|
|
DB::connection(config('database.default'))->transaction(function () use ($amount) {
|
|
|
|
$this->client = Client::withTrashed()->where('id', $this->client->id)->lockForUpdate()->first();
|
|
|
|
$this->client->payment_balance = $amount;
|
|
|
|
$this->client->saveQuietly();
|
|
|
|
}, 2);
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-02-03 11:33:07 +01:00
|
|
|
public function adjustCreditBalance(float $amount)
|
|
|
|
{
|
2022-03-27 08:04:13 +02:00
|
|
|
$this->client->credit_balance += $amount;
|
2020-02-03 11:33:07 +01:00
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2024-01-14 05:05:00 +01:00
|
|
|
public function getCreditBalance(): float
|
2020-10-13 05:25:51 +02:00
|
|
|
{
|
2022-09-12 00:33:59 +02:00
|
|
|
$credits = Credit::withTrashed()->where('client_id', $this->client->id)
|
2020-10-13 05:25:51 +02:00
|
|
|
->where('is_deleted', false)
|
2022-06-21 11:57:17 +02:00
|
|
|
->where(function ($query) {
|
|
|
|
$query->whereDate('due_date', '<=', now()->format('Y-m-d'))
|
2021-07-03 23:48:16 +02:00
|
|
|
->orWhereNull('due_date');
|
|
|
|
})
|
2022-06-21 11:57:17 +02:00
|
|
|
->orderBy('created_at', 'ASC');
|
2020-10-13 05:25:51 +02:00
|
|
|
|
|
|
|
return Number::roundValue($credits->sum('balance'), $this->client->currency()->precision);
|
|
|
|
}
|
|
|
|
|
2022-06-21 11:57:17 +02:00
|
|
|
public function getCredits()
|
2020-10-13 14:28:30 +02:00
|
|
|
{
|
2023-08-06 09:35:19 +02:00
|
|
|
return Credit::query()->where('client_id', $this->client->id)
|
2020-10-13 14:28:30 +02:00
|
|
|
->where('is_deleted', false)
|
|
|
|
->where('balance', '>', 0)
|
2022-06-21 11:57:17 +02:00
|
|
|
->where(function ($query) {
|
|
|
|
$query->whereDate('due_date', '<=', now()->format('Y-m-d'))
|
2021-07-04 01:02:16 +02:00
|
|
|
->orWhereNull('due_date');
|
|
|
|
})
|
2022-06-21 11:57:17 +02:00
|
|
|
->orderBy('created_at', 'ASC')->get();
|
2020-10-13 14:28:30 +02:00
|
|
|
}
|
|
|
|
|
2021-01-18 03:12:48 +01:00
|
|
|
public function getPaymentMethods(float $amount)
|
|
|
|
{
|
|
|
|
return (new PaymentMethod($this->client, $amount))->run();
|
|
|
|
}
|
2020-10-13 14:28:30 +02:00
|
|
|
|
2021-08-01 07:46:40 +02:00
|
|
|
public function merge(Client $mergable_client)
|
|
|
|
{
|
2021-08-01 09:21:08 +02:00
|
|
|
$this->client = (new Merge($this->client, $mergable_client))->run();
|
|
|
|
|
|
|
|
return $this;
|
2021-08-01 07:46:40 +02:00
|
|
|
}
|
|
|
|
|
2021-09-14 13:55:24 +02:00
|
|
|
/**
|
|
|
|
* Generate the client statement.
|
2022-06-21 11:57:17 +02:00
|
|
|
*
|
|
|
|
* @param array $options
|
2023-01-17 01:00:12 +01:00
|
|
|
* @param bool $send_email determines if we should send this statement direct to the client
|
2021-09-14 13:55:24 +02:00
|
|
|
*/
|
2023-01-17 01:00:12 +01:00
|
|
|
public function statement(array $options = [], bool $send_email = false)
|
2021-09-14 13:55:24 +02:00
|
|
|
{
|
2023-01-17 01:00:12 +01:00
|
|
|
$statement = (new Statement($this->client, $options));
|
|
|
|
|
|
|
|
$pdf = $statement->run();
|
|
|
|
|
2023-02-16 02:36:09 +01:00
|
|
|
if ($send_email) {
|
2023-06-13 21:23:54 +02:00
|
|
|
// If selected, ignore clients that don't have any invoices to put on the statement.
|
2023-06-15 19:46:11 +02:00
|
|
|
if (!empty($options['only_clients_with_invoices']) && $statement->getInvoices()->count() == 0) {
|
2023-06-13 21:23:54 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-08-11 06:18:58 +02:00
|
|
|
$this->emailStatement($pdf, $statement->options);
|
|
|
|
return;
|
2023-02-16 02:36:09 +01:00
|
|
|
}
|
2023-01-17 01:00:12 +01:00
|
|
|
|
|
|
|
return $pdf;
|
2021-09-14 13:55:24 +02:00
|
|
|
}
|
|
|
|
|
2023-01-17 01:00:12 +01:00
|
|
|
/**
|
|
|
|
* Emails the statement to the client
|
2023-02-16 02:36:09 +01:00
|
|
|
*
|
2023-01-17 01:00:12 +01:00
|
|
|
* @param mixed $pdf The pdf blob
|
|
|
|
* @param array $options The statement options array
|
|
|
|
*/
|
|
|
|
private function emailStatement($pdf, array $options): void
|
|
|
|
{
|
|
|
|
$this->client_start_date = $this->translateDate($options['start_date'], $this->client->date_format(), $this->client->locale());
|
|
|
|
$this->client_end_date = $this->translateDate($options['end_date'], $this->client->date_format(), $this->client->locale());
|
2024-01-14 05:05:00 +01:00
|
|
|
|
2023-03-07 09:52:37 +01:00
|
|
|
$email_object = $this->buildStatementMailableData($pdf);
|
|
|
|
Email::dispatch($email_object, $this->client->company);
|
2023-01-17 01:00:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Builds and returns an EmailObject for Client Statements
|
2023-02-16 02:36:09 +01:00
|
|
|
*
|
2023-01-17 01:00:12 +01:00
|
|
|
* @param mixed $pdf The PDF to send
|
|
|
|
* @return EmailObject The EmailObject to send
|
|
|
|
*/
|
2024-01-14 05:05:00 +01:00
|
|
|
public function buildStatementMailableData($pdf): EmailObject
|
2023-01-17 01:00:12 +01:00
|
|
|
{
|
2023-05-06 14:19:55 +02:00
|
|
|
$email = $this->client->present()->email();
|
|
|
|
|
2024-01-14 05:05:00 +01:00
|
|
|
$email_object = new EmailObject();
|
2023-05-06 14:19:55 +02:00
|
|
|
$email_object->to = [new Address($email, $this->client->present()->name())];
|
|
|
|
|
|
|
|
$cc_contacts = $this->client
|
|
|
|
->contacts()
|
|
|
|
->where('send_email', true)
|
2023-10-26 04:57:44 +02:00
|
|
|
->where('email', '!=', $email)
|
2023-05-06 14:19:55 +02:00
|
|
|
->get();
|
|
|
|
|
|
|
|
foreach ($cc_contacts as $contact) {
|
2024-01-14 05:05:00 +01:00
|
|
|
|
2023-05-06 14:19:55 +02:00
|
|
|
$email_object->cc[] = new Address($contact->email, $contact->present()->name());
|
2024-01-14 05:05:00 +01:00
|
|
|
|
2023-05-06 14:19:55 +02:00
|
|
|
}
|
|
|
|
|
2023-10-27 08:13:23 +02:00
|
|
|
$invoice = $this->client->invoices()->whereHas('invitations')->first();
|
|
|
|
|
2023-01-17 01:00:12 +01:00
|
|
|
$email_object->attachments = [['file' => base64_encode($pdf), 'name' => ctrans('texts.statement') . ".pdf"]];
|
2023-03-06 09:07:25 +01:00
|
|
|
$email_object->client_id = $this->client->id;
|
2023-10-27 08:13:23 +02:00
|
|
|
$email_object->entity_class = Invoice::class;
|
|
|
|
$email_object->entity_id = $invoice->id ?? null;
|
|
|
|
$email_object->invitation_id = $invoice->invitations->first()->id ?? null;
|
2023-01-17 01:00:12 +01:00
|
|
|
$email_object->email_template_subject = 'email_subject_statement';
|
|
|
|
$email_object->email_template_body = 'email_template_statement';
|
|
|
|
$email_object->variables = [
|
|
|
|
'$client' => $this->client->present()->name(),
|
|
|
|
'$start_date' => $this->client_start_date,
|
|
|
|
'$end_date' => $this->client_end_date,
|
|
|
|
];
|
|
|
|
|
|
|
|
return $email_object;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Saves the client instance
|
2023-02-16 02:36:09 +01:00
|
|
|
*
|
2023-01-17 01:00:12 +01:00
|
|
|
* @return Client The Client Model
|
|
|
|
*/
|
2024-01-14 05:05:00 +01:00
|
|
|
public function save(): Client
|
2020-02-03 11:33:07 +01:00
|
|
|
{
|
2023-02-01 05:00:45 +01:00
|
|
|
$this->client->saveQuietly();
|
2020-02-03 11:33:07 +01:00
|
|
|
|
2022-03-01 11:25:18 +01:00
|
|
|
return $this->client->fresh();
|
2020-02-03 11:33:07 +01:00
|
|
|
}
|
|
|
|
}
|