1
0
mirror of https://github.com/invoiceninja/invoiceninja.git synced 2024-11-15 23:52:33 +01:00
invoiceninja/app/Repositories/ExpenseRepository.php

210 lines
5.5 KiB
PHP
Raw Normal View History

<?php
/**
* Invoice Ninja (https://invoiceninja.com).
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
2024-04-12 06:15:41 +02:00
* @copyright Copyright (c) 2024. Invoice Ninja LLC (https://invoiceninja.com)
*
2021-06-16 08:58:16 +02:00
* @license https://www.elastic.co/licensing/elastic-license
*/
namespace App\Repositories;
2023-11-03 01:07:05 +01:00
use App\Factory\ExpenseFactory;
2023-11-26 08:41:42 +01:00
use App\Jobs\Expense\VendorExpenseNotify;
use App\Libraries\Currency\Conversion\CurrencyApi;
use App\Models\Expense;
2023-11-03 01:07:05 +01:00
use App\Models\ExpenseCategory;
use App\Utils\Traits\GeneratesCounter;
2023-11-03 01:07:05 +01:00
use Carbon\Exceptions\InvalidFormatException;
2023-11-26 08:41:42 +01:00
use Illuminate\Database\Eloquent\Collection;
use Illuminate\Database\QueryException;
use Illuminate\Support\Carbon;
/**
* ExpenseRepository.
*/
class ExpenseRepository extends BaseRepository
{
use GeneratesCounter;
private $completed = true;
2023-11-24 06:10:10 +01:00
private $notify_vendor = false;
/**
* Saves the expense and its contacts.
*
* @param array $data The data
* @param \App\Models\Expense $expense The expense
*
2023-02-16 02:36:09 +01:00
* @return \App\Models\Expense
*/
public function save(array $data, Expense $expense): Expense
{
2023-11-24 06:10:10 +01:00
/** @var \App\Models\User $user */
$user = auth()->user();
2024-04-03 07:03:52 +02:00
$payment_date = &$data['payment_date'];
$vendor_id = &$data['vendor_id'];
if($payment_date && $payment_date == $expense->payment_date) {
2023-11-23 13:54:42 +01:00
//do nothing
2024-04-03 07:03:52 +02:00
} elseif($payment_date && strlen($payment_date) > 1 && $user->company()->notify_vendor_when_paid && ($vendor_id || $expense->vendor_id)) {
2023-11-24 06:10:10 +01:00
$this->notify_vendor = true;
2023-11-23 13:54:42 +01:00
}
$expense->fill($data);
2022-05-11 08:29:56 +02:00
if (!$expense->id) {
2022-05-11 08:29:56 +02:00
$expense = $this->processExchangeRates($data, $expense);
}
2022-05-11 08:29:56 +02:00
2023-02-16 02:36:09 +01:00
if (empty($expense->number)) {
$expense = $this->findAndSaveNumber($expense);
2023-11-26 08:41:42 +01:00
} else {
2023-11-23 13:54:42 +01:00
$expense->saveQuietly();
2023-11-26 08:41:42 +01:00
}
2020-10-12 22:42:02 +02:00
if (array_key_exists('documents', $data)) {
$this->saveDocuments($data['documents'], $expense);
}
2023-11-26 08:41:42 +01:00
if($this->notify_vendor) {
2023-11-24 06:10:10 +01:00
VendorExpenseNotify::dispatch($expense, $expense->company->db);
2023-11-26 08:41:42 +01:00
}
2023-11-24 06:10:10 +01:00
2024-04-03 07:03:52 +02:00
if($payment_date && strlen($payment_date) > 1 && $expense->purchase_order) {
$purchase_order = $expense->purchase_order;
$purchase_order->balance = round($purchase_order->amount - $expense->amount, 2);
$purchase_order->paid_to_date = $expense->amount;
$purchase_order->save();
}
return $expense;
}
/**
* Store expenses in bulk.
*
* @param array $expense
*
2020-11-03 14:34:24 +01:00
* @return \App\Models\Expense|null
*/
public function create($expense): ?Expense
{
2023-10-09 06:16:35 +02:00
/** @var \App\Models\User $user */
$user = auth()->user();
return $this->save(
$expense,
2023-10-09 06:16:35 +02:00
ExpenseFactory::create($user->company()->id, $user->id)
);
}
2022-05-11 08:29:56 +02:00
/**
2023-02-16 02:36:09 +01:00
* @param mixed $data
* @param mixed $expense
* @return Expense
* @throws InvalidFormatException
*/
public function processExchangeRates($data, $expense): Expense
2022-05-11 08:29:56 +02:00
{
if (array_key_exists('exchange_rate', $data) && isset($data['exchange_rate']) && $data['exchange_rate'] != 1) {
2022-05-11 08:29:56 +02:00
return $expense;
}
$expense_currency = $data['currency_id'];
$company_currency = $expense->company->settings->currency_id;
if ($company_currency != $expense_currency) {
$exchange_rate = new CurrencyApi();
$expense->exchange_rate = $exchange_rate->exchangeRate($expense_currency, $company_currency, Carbon::parse($expense->date));
return $expense;
}
2022-05-11 08:29:56 +02:00
return $expense;
}
2024-01-14 05:05:00 +01:00
public function delete($expense): Expense
{
2024-01-14 05:05:00 +01:00
if ($expense->transaction()->exists()) {
2024-01-14 05:05:00 +01:00
$exp_ids = collect(explode(',', $expense->transaction->expense_id))->filter(function ($id) use ($expense) {
return $id != $expense->hashed_id;
})->implode(',');
2024-01-14 05:05:00 +01:00
$expense->transaction_id = null;
$expense->saveQuietly();
$expense->transaction->expense_id = $exp_ids;
2023-08-29 05:40:35 +02:00
if(strlen($exp_ids) <= 2) {
$expense->transaction->status_id = 1;
}
$expense->transaction->saveQuietly();
}
parent::delete($expense);
return $expense;
}
/**
* Handle race conditions when creating expense numbers
*
* @param Expense $expense
* @return \App\Models\Expense
*/
private function findAndSaveNumber($expense): Expense
{
$x = 1;
do {
try {
$expense->number = $this->getNextExpenseNumber($expense);
$expense->saveQuietly();
$this->completed = false;
} catch (QueryException $e) {
$x++;
2023-02-16 02:36:09 +01:00
if ($x > 50) {
$this->completed = false;
2023-02-16 02:36:09 +01:00
}
}
} while ($this->completed);
return $expense;
}
2024-01-14 05:05:00 +01:00
2023-11-03 01:07:05 +01:00
/**
* Categorize Expenses in bulk
*
* @param Collection $expenses
* @param int $category_id
* @return void
*/
2023-11-03 05:14:34 +01:00
public function categorize(Collection $expenses, int $category_id): void
2023-11-03 01:07:05 +01:00
{
$ec = ExpenseCategory::withTrashed()->find($category_id);
2023-11-26 08:41:42 +01:00
$expenses->when($ec)
->each(function ($expense) use ($ec) {
2024-01-14 05:05:00 +01:00
2023-11-26 08:41:42 +01:00
$expense->category_id = $ec->id;
$expense->save();
2023-11-03 01:07:05 +01:00
2023-11-26 08:41:42 +01:00
});
2023-11-03 01:07:05 +01:00
}
}