mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-10-31 09:27:33 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			210 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			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();
 | |
| 
 | |
|                  });
 | |
|     }
 | |
| 
 | |
| }
 |