mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-25 21:16:47 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			244 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			244 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * OperationsController.php
 | |
|  * Copyright (C) 2016 thegrumpydictator@gmail.com
 | |
|  *
 | |
|  * This software may be modified and distributed under the terms of the
 | |
|  * Creative Commons Attribution-ShareAlike 4.0 International License.
 | |
|  *
 | |
|  * See the LICENSE file for details.
 | |
|  */
 | |
| 
 | |
| declare(strict_types = 1);
 | |
| 
 | |
| namespace FireflyIII\Http\Controllers\Report;
 | |
| 
 | |
| 
 | |
| use Carbon\Carbon;
 | |
| use FireflyIII\Helpers\Collector\JournalCollectorInterface;
 | |
| use FireflyIII\Http\Controllers\Controller;
 | |
| use FireflyIII\Models\Transaction;
 | |
| use FireflyIII\Models\TransactionType;
 | |
| use FireflyIII\Support\CacheProperties;
 | |
| use Illuminate\Support\Collection;
 | |
| 
 | |
| /**
 | |
|  * Class OperationsController
 | |
|  *
 | |
|  * @package FireflyIII\Http\Controllers\Report
 | |
|  */
 | |
| class OperationsController extends Controller
 | |
| {
 | |
| 
 | |
|     /**
 | |
|      * @param Collection $accounts
 | |
|      * @param Carbon     $start
 | |
|      * @param Carbon     $end
 | |
|      *
 | |
|      * @return mixed|string
 | |
|      */
 | |
|     public function expenses(Collection $accounts, Carbon $start, Carbon $end)
 | |
|     {
 | |
|         // chart properties for cache:
 | |
|         $cache = new CacheProperties;
 | |
|         $cache->addProperty($start);
 | |
|         $cache->addProperty($end);
 | |
|         $cache->addProperty('expense-report');
 | |
|         $cache->addProperty($accounts->pluck('id')->toArray());
 | |
|         if ($cache->has()) {
 | |
|             return $cache->get();
 | |
|         }
 | |
|         $expenses = $this->getExpenseReport($start, $end, $accounts);
 | |
|         $result   = view('reports.partials.expenses', compact('expenses'))->render();
 | |
|         $cache->store($result);
 | |
| 
 | |
|         return $result;
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Collection $accounts
 | |
|      * @param Carbon     $start
 | |
|      * @param Carbon     $end
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     public function income(Collection $accounts, Carbon $start, Carbon $end)
 | |
|     {
 | |
|         // chart properties for cache:
 | |
|         $cache = new CacheProperties;
 | |
|         $cache->addProperty($start);
 | |
|         $cache->addProperty($end);
 | |
|         $cache->addProperty('income-report');
 | |
|         $cache->addProperty($accounts->pluck('id')->toArray());
 | |
|         if ($cache->has()) {
 | |
|             return $cache->get();
 | |
|         }
 | |
|         $income = $this->getIncomeReport($start, $end, $accounts);
 | |
| 
 | |
|         $result = view('reports.partials.income', compact('income'))->render();
 | |
|         $cache->store($result);
 | |
| 
 | |
|         return $result;
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Collection $accounts
 | |
|      * @param Carbon     $start
 | |
|      * @param Carbon     $end
 | |
|      *
 | |
|      * @return mixed|string
 | |
|      */
 | |
|     public function operations(Collection $accounts, Carbon $start, Carbon $end)
 | |
|     {
 | |
|         // chart properties for cache:
 | |
|         $cache = new CacheProperties;
 | |
|         $cache->addProperty($start);
 | |
|         $cache->addProperty($end);
 | |
|         $cache->addProperty('inc-exp-report');
 | |
|         $cache->addProperty($accounts->pluck('id')->toArray());
 | |
|         if ($cache->has()) {
 | |
|             return $cache->get();
 | |
|         }
 | |
| 
 | |
|         $incomes   = $this->getIncomeReport($start, $end, $accounts);
 | |
|         $expenses  = $this->getExpenseReport($start, $end, $accounts);
 | |
|         $incomeSum = array_sum(
 | |
|             array_map(
 | |
|                 function ($item) {
 | |
|                     return $item['sum'];
 | |
|                 }, $incomes
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         $expensesSum = array_sum(
 | |
|             array_map(
 | |
|                 function ($item) {
 | |
|                     return $item['sum'];
 | |
|                 }, $expenses
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         $result = view('reports.partials.operations', compact('incomeSum', 'expensesSum'))->render();
 | |
|         $cache->store($result);
 | |
| 
 | |
|         return $result;
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Carbon     $start
 | |
|      * @param Carbon     $end
 | |
|      * @param Collection $accounts
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     private function getExpenseReport(Carbon $start, Carbon $end, Collection $accounts): array
 | |
|     {
 | |
|         // get all expenses for the given accounts in the given period!
 | |
|         // also transfers!
 | |
|         // get all transactions:
 | |
|         /** @var JournalCollectorInterface $collector */
 | |
|         $collector = app(JournalCollectorInterface::class);
 | |
|         $collector->setAccounts($accounts)->setRange($start, $end);
 | |
|         $collector->setTypes([TransactionType::WITHDRAWAL, TransactionType::TRANSFER])
 | |
|                   ->withOpposingAccount()
 | |
|                   ->enableInternalFilter();
 | |
|         $transactions = $collector->getJournals();
 | |
|         $transactions = $transactions->filter(
 | |
|             function (Transaction $transaction) {
 | |
|                 // return negative amounts only.
 | |
|                 if (bccomp($transaction->transaction_amount, '0') === -1) {
 | |
|                     return $transaction;
 | |
|                 }
 | |
| 
 | |
|                 return false;
 | |
|             }
 | |
|         );
 | |
|         $expenses     = $this->groupByOpposing($transactions);
 | |
| 
 | |
|         // sort the result
 | |
|         // Obtain a list of columns
 | |
|         $sum = [];
 | |
|         foreach ($expenses as $accountId => $row) {
 | |
|             $sum[$accountId] = floatval($row['sum']);
 | |
|         }
 | |
| 
 | |
|         array_multisort($sum, SORT_ASC, $expenses);
 | |
| 
 | |
|         return $expenses;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Carbon     $start
 | |
|      * @param Carbon     $end
 | |
|      * @param Collection $accounts
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     private function getIncomeReport(Carbon $start, Carbon $end, Collection $accounts): array
 | |
|     {
 | |
|         // get all expenses for the given accounts in the given period!
 | |
|         // also transfers!
 | |
|         // get all transactions:
 | |
|         /** @var JournalCollectorInterface $collector */
 | |
|         $collector = app(JournalCollectorInterface::class);
 | |
|         $collector->setAccounts($accounts)->setRange($start, $end);
 | |
|         $collector->setTypes([TransactionType::DEPOSIT, TransactionType::TRANSFER])
 | |
|                   ->withOpposingAccount()
 | |
|                   ->enableInternalFilter();
 | |
|         $transactions = $collector->getJournals();
 | |
|         $transactions = $transactions->filter(
 | |
|             function (Transaction $transaction) {
 | |
|                 // return positive amounts only.
 | |
|                 if (bccomp($transaction->transaction_amount, '0') === 1) {
 | |
|                     return $transaction;
 | |
|                 }
 | |
| 
 | |
|                 return false;
 | |
|             }
 | |
|         );
 | |
|         $income       = $this->groupByOpposing($transactions);
 | |
| 
 | |
|         // sort the result
 | |
|         // Obtain a list of columns
 | |
|         $sum = [];
 | |
|         foreach ($income as $accountId => $row) {
 | |
|             $sum[$accountId] = floatval($row['sum']);
 | |
|         }
 | |
| 
 | |
|         array_multisort($sum, SORT_DESC, $income);
 | |
| 
 | |
|         return $income;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Collection $transactions
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     private function groupByOpposing(Collection $transactions): array
 | |
|     {
 | |
|         $expenses = [];
 | |
|         // join the result together:
 | |
|         foreach ($transactions as $transaction) {
 | |
|             $opposingId = $transaction->opposing_account_id;
 | |
|             $name       = $transaction->opposing_account_name;
 | |
|             if (!isset($expenses[$opposingId])) {
 | |
|                 $expenses[$opposingId] = [
 | |
|                     'id'    => $opposingId,
 | |
|                     'name'  => $name,
 | |
|                     'sum'   => '0',
 | |
|                     'count' => 0,
 | |
|                 ];
 | |
|             }
 | |
|             $expenses[$opposingId]['sum'] = bcadd($expenses[$opposingId]['sum'], $transaction->transaction_amount);
 | |
|             $expenses[$opposingId]['count']++;
 | |
|         }
 | |
| 
 | |
| 
 | |
|         return $expenses;
 | |
|     }
 | |
| 
 | |
| } |