mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-11-04 08:17:32 -05: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();
 | 
						|
 | 
						|
                 });
 | 
						|
    }
 | 
						|
 | 
						|
}
 |