| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  | <?php | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * AugumentData.php | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * Copyright (c) 2019 thegrumpydictator@gmail.com | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This file is part of Firefly III (https://github.com/firefly-iii). | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This program is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU Affero General Public License as | 
					
						
							|  |  |  |  * published by the Free Software Foundation, either version 3 of the | 
					
						
							|  |  |  |  * License, or (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * GNU Affero General Public License for more details. | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * You should have received a copy of the GNU Affero General Public License | 
					
						
							|  |  |  |  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | declare(strict_types=1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Support\Http\Controllers; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use Carbon\Carbon; | 
					
						
							| 
									
										
										
										
											2019-05-30 12:31:19 +02:00
										 |  |  | use FireflyIII\Helpers\Collector\GroupCollectorInterface; | 
					
						
							| 
									
										
										
										
											2018-08-11 14:06:49 +02:00
										 |  |  | use FireflyIII\Models\Account; | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  | use FireflyIII\Models\AccountType; | 
					
						
							|  |  |  | use FireflyIII\Models\Budget; | 
					
						
							|  |  |  | use FireflyIII\Models\BudgetLimit; | 
					
						
							| 
									
										
										
										
											2018-12-31 08:11:57 +01:00
										 |  |  | use FireflyIII\Models\TransactionType; | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  | use FireflyIII\Repositories\Account\AccountRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2019-08-30 08:09:39 +02:00
										 |  |  | use FireflyIII\Repositories\Budget\BudgetLimitRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  | use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2019-08-29 21:42:55 +02:00
										 |  |  | use FireflyIII\Repositories\Budget\OperationsRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  | use FireflyIII\Repositories\Category\CategoryRepositoryInterface; | 
					
						
							| 
									
										
										
										
											2018-12-31 07:58:13 +01:00
										 |  |  | use FireflyIII\Support\CacheProperties; | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  | use Illuminate\Support\Collection; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Trait AugumentData | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | trait AugumentData | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-08-11 14:06:49 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Searches for the opposing account. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Collection $accounts | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function combineAccounts(Collection $accounts): array // filter + group data
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var AccountRepositoryInterface $repository */ | 
					
						
							|  |  |  |         $repository = app(AccountRepositoryInterface::class); | 
					
						
							|  |  |  |         $combined   = []; | 
					
						
							|  |  |  |         /** @var Account $expenseAccount */ | 
					
						
							|  |  |  |         foreach ($accounts as $expenseAccount) { | 
					
						
							|  |  |  |             $collection = new Collection; | 
					
						
							|  |  |  |             $collection->push($expenseAccount); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $revenue = $repository->findByName($expenseAccount->name, [AccountType::REVENUE]); | 
					
						
							|  |  |  |             if (null !== $revenue) { | 
					
						
							|  |  |  |                 $collection->push($revenue); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             $combined[$expenseAccount->name] = $collection; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $combined; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-31 07:58:13 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Small helper function for the revenue and expense account charts. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param array $names | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function expandNames(array $names): array | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $result = []; | 
					
						
							|  |  |  |         foreach ($names as $entry) { | 
					
						
							|  |  |  |             $result[$entry['name']] = 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Small helper function for the revenue and expense account charts. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Collection $accounts | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function extractNames(Collection $accounts): array | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $return = []; | 
					
						
							|  |  |  |         /** @var Account $account */ | 
					
						
							|  |  |  |         foreach ($accounts as $account) { | 
					
						
							|  |  |  |             $return[$account->id] = $account->name; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get the account names belonging to a bunch of account ID's. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param array $accountIds | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function getAccountNames(array $accountIds): array // extract info from array.
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var AccountRepositoryInterface $repository */ | 
					
						
							|  |  |  |         $repository = app(AccountRepositoryInterface::class); | 
					
						
							|  |  |  |         $accounts   = $repository->getAccountsByType([AccountType::ASSET, AccountType::DEFAULT, AccountType::EXPENSE, AccountType::CASH]); | 
					
						
							|  |  |  |         $grouped    = $accounts->groupBy('id')->toArray(); | 
					
						
							|  |  |  |         $return     = []; | 
					
						
							|  |  |  |         foreach ($accountIds as $accountId) { | 
					
						
							|  |  |  |             if (isset($grouped[$accountId])) { | 
					
						
							|  |  |  |                 $return[$accountId] = $grouped[$accountId][0]['name']; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $return[0] = '(no name)'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Get the budget names from a set of budget ID's. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param array $budgetIds | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function getBudgetNames(array $budgetIds): array // extract info from array.
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var BudgetRepositoryInterface $repository */ | 
					
						
							|  |  |  |         $repository = app(BudgetRepositoryInterface::class); | 
					
						
							|  |  |  |         $budgets    = $repository->getBudgets(); | 
					
						
							|  |  |  |         $grouped    = $budgets->groupBy('id')->toArray(); | 
					
						
							|  |  |  |         $return     = []; | 
					
						
							|  |  |  |         foreach ($budgetIds as $budgetId) { | 
					
						
							|  |  |  |             if (isset($grouped[$budgetId])) { | 
					
						
							|  |  |  |                 $return[$budgetId] = $grouped[$budgetId][0]['name']; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $return[0] = (string)trans('firefly.no_budget'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Get the category names from a set of category ID's. Small helper function for some of the charts. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param array $categoryIds | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function getCategoryNames(array $categoryIds): array // extract info from array.
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         /** @var CategoryRepositoryInterface $repository */ | 
					
						
							|  |  |  |         $repository = app(CategoryRepositoryInterface::class); | 
					
						
							|  |  |  |         $categories = $repository->getCategories(); | 
					
						
							|  |  |  |         $grouped    = $categories->groupBy('id')->toArray(); | 
					
						
							|  |  |  |         $return     = []; | 
					
						
							|  |  |  |         foreach ($categoryIds as $categoryId) { | 
					
						
							|  |  |  |             if (isset($grouped[$categoryId])) { | 
					
						
							|  |  |  |                 $return[$categoryId] = $grouped[$categoryId][0]['name']; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-08-27 18:59:30 +02:00
										 |  |  |         $return[0] = (string)trans('firefly.no_category'); | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return $return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-31 07:58:13 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Gets all budget limits for a budget. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Budget $budget | 
					
						
							|  |  |  |      * @param Carbon $start | 
					
						
							|  |  |  |      * @param Carbon $end | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function getLimits(Budget $budget, Carbon $start, Carbon $end): Collection // get data + augment with info
 | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-08-29 21:42:55 +02:00
										 |  |  |         /** @var OperationsRepositoryInterface $opsRepository */ | 
					
						
							|  |  |  |         $opsRepository = app(OperationsRepositoryInterface::class); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-30 08:09:39 +02:00
										 |  |  |         /** @var BudgetLimitRepositoryInterface $blRepository */ | 
					
						
							|  |  |  |         $blRepository = app(BudgetLimitRepositoryInterface::class); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-31 07:58:13 +01:00
										 |  |  |         // properties for cache
 | 
					
						
							|  |  |  |         $cache = new CacheProperties; | 
					
						
							|  |  |  |         $cache->addProperty($start); | 
					
						
							|  |  |  |         $cache->addProperty($end); | 
					
						
							|  |  |  |         $cache->addProperty($budget->id); | 
					
						
							|  |  |  |         $cache->addProperty('get-limits'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($cache->has()) { | 
					
						
							|  |  |  |             return $cache->get(); // @codeCoverageIgnore
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-30 08:09:39 +02:00
										 |  |  |         $set    = $blRepository->getBudgetLimits($budget, $start, $end); | 
					
						
							| 
									
										
										
										
											2018-12-31 07:58:13 +01:00
										 |  |  |         $limits = new Collection(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /** @var BudgetLimit $entry */ | 
					
						
							|  |  |  |         foreach ($set as $entry) { | 
					
						
							| 
									
										
										
										
											2019-08-29 21:42:55 +02:00
										 |  |  |             $entry->spent = $opsRepository->spentInPeriod(new Collection([$budget]), new Collection(), $entry->start_date, $entry->end_date); | 
					
						
							| 
									
										
										
										
											2018-12-31 07:58:13 +01:00
										 |  |  |             $limits->push($entry); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $cache->store($limits); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $set; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2018-08-11 14:06:49 +02:00
										 |  |  |      * Group set of transactions by name of opposing account. | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |      * @param array $array | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2018-08-11 14:06:49 +02:00
										 |  |  |      * @return array | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |     protected function groupByName(array $array): array // filter + group data
 | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 14:06:49 +02:00
										 |  |  |         // group by opposing account name.
 | 
					
						
							|  |  |  |         $grouped = []; | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         /** @var array $journal */ | 
					
						
							|  |  |  |         foreach ($array as $journal) { | 
					
						
							|  |  |  |             $name = '(no name)'; | 
					
						
							|  |  |  |             if (TransactionType::WITHDRAWAL === $journal['transaction_type_type']) { | 
					
						
							|  |  |  |                 $name = $journal['destination_account_name']; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (TransactionType::WITHDRAWAL !== $journal['transaction_type_type']) { | 
					
						
							|  |  |  |                 $name = $journal['source_account_name']; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 14:06:49 +02:00
										 |  |  |             $grouped[$name] = $grouped[$name] ?? '0'; | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |             $grouped[$name] = bcadd($journal['amount'], $grouped[$name]); | 
					
						
							| 
									
										
										
										
											2018-08-11 14:06:49 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-08-11 06:39:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 14:06:49 +02:00
										 |  |  |         return $grouped; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-31 08:11:57 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Spent in a period. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param Collection $assets | 
					
						
							|  |  |  |      * @param Collection $opposing | 
					
						
							| 
									
										
										
										
											2019-08-30 08:09:39 +02:00
										 |  |  |      * @param Carbon     $start | 
					
						
							|  |  |  |      * @param Carbon     $end | 
					
						
							| 
									
										
										
										
											2018-12-31 08:11:57 +01:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return array | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function spentInPeriod(Collection $assets, Collection $opposing, Carbon $start, Carbon $end): array // get data + augment with info
 | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         /** @var GroupCollectorInterface $collector */ | 
					
						
							|  |  |  |         $collector = app(GroupCollectorInterface::class); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $total = $assets->merge($opposing); | 
					
						
							|  |  |  |         $collector->setRange($start, $end)->setTypes([TransactionType::WITHDRAWAL])->setAccounts($total); | 
					
						
							|  |  |  |         $journals = $collector->getExtractedJournals(); | 
					
						
							|  |  |  |         $sum      = [ | 
					
						
							| 
									
										
										
										
											2018-12-31 08:11:57 +01:00
										 |  |  |             'grand_sum'    => '0', | 
					
						
							|  |  |  |             'per_currency' => [], | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  |         // loop to support multi currency
 | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |         foreach ($journals as $journal) { | 
					
						
							|  |  |  |             $currencyId = (int)$journal['currency_id']; | 
					
						
							| 
									
										
										
										
											2018-12-31 08:11:57 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // if not set, set to zero:
 | 
					
						
							|  |  |  |             if (!isset($sum['per_currency'][$currencyId])) { | 
					
						
							|  |  |  |                 $sum['per_currency'][$currencyId] = [ | 
					
						
							|  |  |  |                     'sum'      => '0', | 
					
						
							|  |  |  |                     'currency' => [ | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |                         'name'           => $journal['currency_name'], | 
					
						
							|  |  |  |                         'symbol'         => $journal['currency_symbol'], | 
					
						
							|  |  |  |                         'decimal_places' => $journal['currency_decimal_places'], | 
					
						
							| 
									
										
										
										
											2018-12-31 08:11:57 +01:00
										 |  |  |                     ], | 
					
						
							|  |  |  |                 ]; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // add amount
 | 
					
						
							| 
									
										
										
										
											2019-05-31 13:35:33 +02:00
										 |  |  |             $sum['per_currency'][$currencyId]['sum'] = bcadd($sum['per_currency'][$currencyId]['sum'], $journal['amount']); | 
					
						
							|  |  |  |             $sum['grand_sum']                        = bcadd($sum['grand_sum'], $journal['amount']); | 
					
						
							| 
									
										
										
										
											2018-12-31 08:11:57 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $sum; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-31 07:58:13 +01:00
										 |  |  | } |