mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-31 18:54:58 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			154 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			154 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| /**
 | |
|  * AccountController.php
 | |
|  * Copyright (c) 2020 james@firefly-iii.org
 | |
|  *
 | |
|  * This file is part of Firefly III (https://github.com/firefly-iii).
 | |
|  *
 | |
|  * 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.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU Affero General Public License for more details.
 | |
|  *
 | |
|  * 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/>.
 | |
|  */
 | |
| 
 | |
| declare(strict_types=1);
 | |
| 
 | |
| namespace FireflyIII\Api\V1\Controllers\Autocomplete;
 | |
| 
 | |
| use FireflyIII\Api\V1\Controllers\Controller;
 | |
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteApiRequest;
 | |
| use FireflyIII\Enums\AccountTypeEnum;
 | |
| use FireflyIII\Enums\UserRoleEnum;
 | |
| use FireflyIII\Exceptions\FireflyException;
 | |
| use FireflyIII\Models\Account;
 | |
| use FireflyIII\Repositories\Account\AccountRepositoryInterface;
 | |
| use FireflyIII\Support\Debug\Timer;
 | |
| use FireflyIII\Support\Facades\Amount;
 | |
| use FireflyIII\Support\Facades\Steam;
 | |
| use FireflyIII\Support\Http\Api\AccountFilter;
 | |
| use Illuminate\Http\JsonResponse;
 | |
| use Illuminate\Support\Facades\Log;
 | |
| 
 | |
| /**
 | |
|  * Class AccountController
 | |
|  */
 | |
| class AccountController extends Controller
 | |
| {
 | |
|     use AccountFilter;
 | |
| 
 | |
|     // this array only exists to test if the constructor will use it properly.
 | |
|     protected array $accepts       = ['application/json', 'application/vnd.api+json'];
 | |
|     protected array $acceptedRoles = [UserRoleEnum::READ_ONLY];
 | |
| 
 | |
|     /** @var array<int, string> */
 | |
|     private array                      $balanceTypes;
 | |
|     private AccountRepositoryInterface $repository;
 | |
| 
 | |
|     /**
 | |
|      * AccountController constructor.
 | |
|      */
 | |
|     public function __construct()
 | |
|     {
 | |
|         parent::__construct();
 | |
|         $this->middleware(
 | |
|             function ($request, $next) {
 | |
|                 $this->validateUserGroup($request);
 | |
|                 $this->repository = app(AccountRepositoryInterface::class);
 | |
|                 $this->repository->setUser($this->user);
 | |
|                 $this->repository->setUserGroup($this->userGroup);
 | |
| 
 | |
|                 return $next($request);
 | |
|             }
 | |
|         );
 | |
|         $this->balanceTypes = [AccountTypeEnum::ASSET->value, AccountTypeEnum::LOAN->value, AccountTypeEnum::DEBT->value, AccountTypeEnum::MORTGAGE->value];
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Documentation for this endpoint:
 | |
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/autocomplete/getAccountsAC
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function accounts(AutocompleteApiRequest $request): JsonResponse
 | |
|     {
 | |
|         [
 | |
|             'types' => $types,
 | |
|             'query' => $query,
 | |
|             'date'  => $date,
 | |
|             'limit' => $limit,
 | |
|         ]
 | |
|                      = $request->attributes->all();
 | |
| 
 | |
| 
 | |
|         $date ??= today(config('app.timezone'));
 | |
| 
 | |
|         // set date to end-of-day for account balance. so it is at $date 23:59:59
 | |
|         $date->endOfDay();
 | |
| 
 | |
|         $return      = [];
 | |
|         $timer       = Timer::getInstance();
 | |
|         $timer->start(sprintf('AC accounts "%s"', $query));
 | |
|         $result      = $this->repository->searchAccount((string)$query, $types, $limit);
 | |
|         $allBalances = Steam::accountsBalancesOptimized($result, $date, $this->primaryCurrency, $this->convertToPrimary);
 | |
| 
 | |
|         /** @var Account $account */
 | |
|         foreach ($result as $account) {
 | |
|             $nameWithBalance = $account->name;
 | |
|             $currency        = $this->repository->getAccountCurrency($account) ?? $this->primaryCurrency;
 | |
|             $useCurrency     = $currency;
 | |
|             if (in_array($account->accountType->type, $this->balanceTypes, true)) {
 | |
|                 // this one is correct.
 | |
|                 Log::debug(sprintf('accounts: Call finalAccountBalance with date/time "%s"', $date->toIso8601String()));
 | |
|                 $balance         = $allBalances[$account->id] ?? [];
 | |
|                 $key             = $this->convertToPrimary && $currency->id !== $this->primaryCurrency->id ? 'pc_balance' : 'balance';
 | |
|                 $useCurrency     = $this->convertToPrimary && $currency->id !== $this->primaryCurrency->id ? $this->primaryCurrency : $currency;
 | |
|                 $amount          = $balance[$key] ?? '0';
 | |
|                 $nameWithBalance = sprintf('%s (%s)', $account->name, Amount::formatAnything($useCurrency, $amount, false));
 | |
|             }
 | |
| 
 | |
|             $return[]        = [
 | |
|                 'id'                              => (string)$account->id,
 | |
|                 'name'                            => $account->name,
 | |
|                 'name_with_balance'               => $nameWithBalance,
 | |
|                 'active'                          => $account->active,
 | |
|                 'type'                            => $account->accountType->type,
 | |
|                 'currency_id'                     => (string)$useCurrency->id,
 | |
|                 'currency_name'                   => $useCurrency->name,
 | |
|                 'currency_code'                   => $useCurrency->code,
 | |
|                 'currency_symbol'                 => $useCurrency->symbol,
 | |
|                 'currency_decimal_places'         => $useCurrency->decimal_places,
 | |
|                 'account_currency_id'             => (string)$currency->id,
 | |
|                 'account_currency_name'           => $currency->name,
 | |
|                 'account_currency_code'           => $currency->code,
 | |
|                 'account_currency_symbol'         => $currency->symbol,
 | |
|                 'account_currency_decimal_places' => $currency->decimal_places,
 | |
|             ];
 | |
|         }
 | |
| 
 | |
|         // custom order.
 | |
|         usort(
 | |
|             $return,
 | |
|             static function (array $left, array $right) {
 | |
|                 $order = [AccountTypeEnum::ASSET->value, AccountTypeEnum::REVENUE->value, AccountTypeEnum::EXPENSE->value];
 | |
|                 $posA  = (int)array_search($left['type'], $order, true);
 | |
|                 $posB  = (int)array_search($right['type'], $order, true);
 | |
| 
 | |
|                 return $posA - $posB;
 | |
|             }
 | |
|         );
 | |
|         $timer->stop(sprintf('AC accounts "%s"', $query));
 | |
| 
 | |
|         return response()->api($return);
 | |
|     }
 | |
| }
 |