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

210 lines
5.5 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\Repositories;
use App\Factory\ExpenseFactory;
use App\Jobs\Expense\VendorExpenseNotify;
use App\Libraries\Currency\Conversion\CurrencyApi;
use App\Models\Expense;
use App\Models\ExpenseCategory;
use App\Utils\Traits\GeneratesCounter;
use Carbon\Exceptions\InvalidFormatException;
use Illuminate\Database\Eloquent\Collection;
use Illuminate\Database\QueryException;
use Illuminate\Support\Carbon;
/**
* ExpenseRepository.
*/
class ExpenseRepository extends BaseRepository
{
use GeneratesCounter;
private $completed = true;
private $notify_vendor = false;
/**
* Saves the expense and its contacts.
*
* @param array $data The data
* @param \App\Models\Expense $expense The expense
*
* @return \App\Models\Expense
*/
public function save(array $data, Expense $expense): Expense
{
/** @var \App\Models\User $user */
$user = auth()->user();
$payment_date = &$data['payment_date'];
$vendor_id = &$data['vendor_id'];
if($payment_date && $payment_date == $expense->payment_date) {
//do nothing
} elseif($payment_date && strlen($payment_date) > 1 && $user->company()->notify_vendor_when_paid && ($vendor_id || $expense->vendor_id)) {
$this->notify_vendor = true;
}
$expense->fill($data);
if (!$expense->id) {
$expense = $this->processExchangeRates($data, $expense);
}
if (empty($expense->number)) {
$expense = $this->findAndSaveNumber($expense);
} else {
$expense->saveQuietly();
}
if (array_key_exists('documents', $data)) {
$this->saveDocuments($data['documents'], $expense);
}
if($this->notify_vendor) {
VendorExpenseNotify::dispatch($expense, $expense->company->db);
}
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
*
* @return \App\Models\Expense|null
*/
public function create($expense): ?Expense
{
/** @var \App\Models\User $user */
$user = auth()->user();
return $this->save(
$expense,
ExpenseFactory::create($user->company()->id, $user->id)
);
}
/**
* @param mixed $data
* @param mixed $expense
* @return Expense
* @throws InvalidFormatException
*/
public function processExchangeRates($data, $expense): Expense
{
if (array_key_exists('exchange_rate', $data) && isset($data['exchange_rate']) && $data['exchange_rate'] != 1) {
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;
}
return $expense;
}
public function delete($expense): Expense
{
if ($expense->transaction()->exists()) {
$exp_ids = collect(explode(',', $expense->transaction->expense_id))->filter(function ($id) use ($expense) {
return $id != $expense->hashed_id;
})->implode(',');
$expense->transaction_id = null;
$expense->saveQuietly();
$expense->transaction->expense_id = $exp_ids;
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++;
if ($x > 50) {
$this->completed = false;
}
}
} while ($this->completed);
return $expense;
}
/**
* Categorize Expenses in bulk
*
* @param Collection $expenses
* @param int $category_id
* @return void
*/
public function categorize(Collection $expenses, int $category_id): void
{
$ec = ExpenseCategory::withTrashed()->find($category_id);
$expenses->when($ec)
->each(function ($expense) use ($ec) {
$expense->category_id = $ec->id;
$expense->save();
});
}
}