2020-01-20 05:53:40 +01:00
|
|
|
<?php
|
2022-09-07 23:57:30 +02:00
|
|
|
|
2020-01-20 05:53:40 +01:00
|
|
|
/**
|
2020-09-06 11:38:10 +02:00
|
|
|
* Invoice Ninja (https://invoiceninja.com).
|
2020-01-20 05:53:40 +01: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-01-20 05:53:40 +01:00
|
|
|
*
|
2021-06-16 08:58:16 +02:00
|
|
|
* @license https://www.elastic.co/licensing/elastic-license
|
2020-01-20 05:53:40 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
namespace App\Repositories;
|
|
|
|
|
|
|
|
use App\Factory\ExpenseFactory;
|
2022-05-11 08:29:56 +02:00
|
|
|
use App\Libraries\Currency\Conversion\CurrencyApi;
|
2020-01-20 05:53:40 +01:00
|
|
|
use App\Models\Expense;
|
|
|
|
use App\Utils\Traits\GeneratesCounter;
|
2022-12-15 23:38:02 +01:00
|
|
|
use Carbon\Exceptions\InvalidFormatException;
|
2022-09-07 23:57:30 +02:00
|
|
|
use Illuminate\Database\QueryException;
|
2023-02-16 02:36:09 +01:00
|
|
|
use Illuminate\Support\Carbon;
|
2020-01-20 05:53:40 +01:00
|
|
|
|
|
|
|
/**
|
2020-09-06 11:38:10 +02:00
|
|
|
* ExpenseRepository.
|
2020-01-20 05:53:40 +01:00
|
|
|
*/
|
|
|
|
class ExpenseRepository extends BaseRepository
|
|
|
|
{
|
|
|
|
use GeneratesCounter;
|
|
|
|
|
2022-09-07 23:57:30 +02:00
|
|
|
private $completed = true;
|
|
|
|
|
2020-01-20 05:53:40 +01:00
|
|
|
/**
|
2020-09-06 11:38:10 +02:00
|
|
|
* Saves the expense and its contacts.
|
2020-01-20 05:53:40 +01:00
|
|
|
*
|
2022-12-15 23:38:02 +01:00
|
|
|
* @param array $data The data
|
|
|
|
* @param \App\Models\Expense $expense The expense
|
2020-01-20 05:53:40 +01:00
|
|
|
*
|
2023-02-16 02:36:09 +01:00
|
|
|
* @return \App\Models\Expense
|
2020-01-20 05:53:40 +01:00
|
|
|
*/
|
2022-12-15 23:38:02 +01:00
|
|
|
public function save(array $data, Expense $expense): Expense
|
2020-01-20 05:53:40 +01:00
|
|
|
{
|
|
|
|
$expense->fill($data);
|
2022-05-11 08:29:56 +02:00
|
|
|
|
2022-09-07 23:57:30 +02:00
|
|
|
if (!$expense->id) {
|
2022-05-11 08:29:56 +02:00
|
|
|
$expense = $this->processExchangeRates($data, $expense);
|
2022-06-21 11:57:17 +02:00
|
|
|
}
|
2022-05-11 08:29:56 +02:00
|
|
|
|
2023-02-16 02:36:09 +01:00
|
|
|
if (empty($expense->number)) {
|
2022-09-07 23:57:30 +02:00
|
|
|
$expense = $this->findAndSaveNumber($expense);
|
2023-02-16 02:36:09 +01:00
|
|
|
}
|
2022-09-07 23:57:30 +02:00
|
|
|
|
2023-02-01 04:12:44 +01:00
|
|
|
$expense->saveQuietly();
|
2020-01-20 05:53:40 +01:00
|
|
|
|
2020-10-12 22:42:02 +02:00
|
|
|
if (array_key_exists('documents', $data)) {
|
|
|
|
$this->saveDocuments($data['documents'], $expense);
|
|
|
|
}
|
|
|
|
|
2020-01-20 05:53:40 +01:00
|
|
|
return $expense;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Store expenses in bulk.
|
|
|
|
*
|
|
|
|
* @param array $expense
|
2022-09-07 23:57:30 +02:00
|
|
|
*
|
2020-11-03 14:34:24 +01:00
|
|
|
* @return \App\Models\Expense|null
|
2020-01-20 05:53:40 +01:00
|
|
|
*/
|
|
|
|
public function create($expense): ?Expense
|
|
|
|
{
|
|
|
|
return $this->save(
|
|
|
|
$expense,
|
|
|
|
ExpenseFactory::create(auth()->user()->company()->id, auth()->user()->id)
|
|
|
|
);
|
|
|
|
}
|
2022-05-11 08:29:56 +02:00
|
|
|
|
2022-12-15 23:38:02 +01:00
|
|
|
/**
|
2023-02-16 02:36:09 +01:00
|
|
|
* @param mixed $data
|
|
|
|
* @param mixed $expense
|
|
|
|
* @return Expense
|
|
|
|
* @throws InvalidFormatException
|
2022-12-15 23:38:02 +01:00
|
|
|
*/
|
2022-09-07 23:57:30 +02:00
|
|
|
public function processExchangeRates($data, $expense): Expense
|
2022-05-11 08:29:56 +02:00
|
|
|
{
|
2022-06-21 11:57:17 +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-06-21 11:57:17 +02:00
|
|
|
|
2022-05-11 08:29:56 +02:00
|
|
|
return $expense;
|
|
|
|
}
|
2022-09-07 23:57:30 +02:00
|
|
|
|
2023-04-05 06:53:48 +02:00
|
|
|
|
|
|
|
public function delete($expense) :Expense
|
|
|
|
{
|
2023-04-05 08:13:42 +02:00
|
|
|
|
2023-04-18 10:29:14 +02:00
|
|
|
if ($expense->transaction()->exists()) {
|
2023-04-05 08:13:42 +02:00
|
|
|
|
|
|
|
$exp_ids = collect(explode(',', $expense->transaction->expense_id))->filter(function ($id) use ($expense) {
|
|
|
|
return $id != $expense->hashed_id;
|
|
|
|
})->implode(',');
|
|
|
|
|
2023-04-05 06:53:48 +02:00
|
|
|
$expense->transaction_id = null;
|
|
|
|
$expense->saveQuietly();
|
2023-04-05 08:13:42 +02:00
|
|
|
|
|
|
|
$expense->transaction->expense_id = $exp_ids;
|
|
|
|
$expense->transaction->saveQuietly();
|
|
|
|
|
2023-04-05 06:53:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
parent::delete($expense);
|
|
|
|
|
|
|
|
return $expense;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-07 23:57:30 +02:00
|
|
|
/**
|
|
|
|
* 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) {
|
2022-09-07 23:57:30 +02:00
|
|
|
$this->completed = false;
|
2023-02-16 02:36:09 +01:00
|
|
|
}
|
2022-09-07 23:57:30 +02:00
|
|
|
}
|
|
|
|
} while ($this->completed);
|
|
|
|
|
|
|
|
return $expense;
|
|
|
|
}
|
2020-01-20 05:53:40 +01:00
|
|
|
}
|