mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-26 05:26:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			334 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			334 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * Search.php
 | |
|  * Copyright (c) 2019 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\Support\Search;
 | |
| 
 | |
| use Carbon\Carbon;
 | |
| use FireflyIII\Helpers\Collector\GroupCollectorInterface;
 | |
| use FireflyIII\Models\AccountType;
 | |
| use FireflyIII\Repositories\Account\AccountRepositoryInterface;
 | |
| use FireflyIII\Repositories\Bill\BillRepositoryInterface;
 | |
| use FireflyIII\Repositories\Budget\BudgetRepositoryInterface;
 | |
| use FireflyIII\Repositories\Category\CategoryRepositoryInterface;
 | |
| use FireflyIII\Repositories\Tag\TagRepositoryInterface;
 | |
| use FireflyIII\User;
 | |
| use Illuminate\Pagination\LengthAwarePaginator;
 | |
| use Illuminate\Support\Collection;
 | |
| use Log;
 | |
| 
 | |
| /**
 | |
|  * Class Search.
 | |
|  */
 | |
| class Search implements SearchInterface
 | |
| {
 | |
|     /** @var AccountRepositoryInterface */
 | |
|     private $accountRepository;
 | |
|     /** @var BillRepositoryInterface */
 | |
|     private $billRepository;
 | |
|     /** @var BudgetRepositoryInterface */
 | |
|     private $budgetRepository;
 | |
|     /** @var CategoryRepositoryInterface */
 | |
|     private $categoryRepository;
 | |
|     /** @var Collection */
 | |
|     private $modifiers;
 | |
|     /** @var string */
 | |
|     private $originalQuery = '';
 | |
|     /** @var float */
 | |
|     private $startTime;
 | |
|     /** @var TagRepositoryInterface */
 | |
|     private $tagRepository;
 | |
|     /** @var User */
 | |
|     private $user;
 | |
|     /** @var array */
 | |
|     private $validModifiers;
 | |
|     /** @var array */
 | |
|     private $words = [];
 | |
|     /** @var int */
 | |
|     private $page;
 | |
| 
 | |
|     /**
 | |
|      * Search constructor.
 | |
|      */
 | |
|     public function __construct()
 | |
|     {
 | |
|         $this->page               = 1;
 | |
|         $this->modifiers          = new Collection;
 | |
|         $this->validModifiers     = (array)config('firefly.search_modifiers');
 | |
|         $this->startTime          = microtime(true);
 | |
|         $this->accountRepository  = app(AccountRepositoryInterface::class);
 | |
|         $this->categoryRepository = app(CategoryRepositoryInterface::class);
 | |
|         $this->budgetRepository   = app(BudgetRepositoryInterface::class);
 | |
|         $this->billRepository     = app(BillRepositoryInterface::class);
 | |
|         $this->tagRepository      = app(TagRepositoryInterface::class);
 | |
| 
 | |
|         if ('testing' === config('app.env')) {
 | |
|             Log::warning(sprintf('%s should not be instantiated in the TEST environment!', get_class($this)));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return Collection
 | |
|      */
 | |
|     public function getModifiers(): Collection
 | |
|     {
 | |
|         return $this->modifiers;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return string
 | |
|      */
 | |
|     public function getWordsAsString(): string
 | |
|     {
 | |
|         $string = implode(' ', $this->words);
 | |
|         if ('' === $string) {
 | |
|             return is_string($this->originalQuery) ? $this->originalQuery : '';
 | |
|         }
 | |
| 
 | |
|         return $string;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return bool
 | |
|      */
 | |
|     public function hasModifiers(): bool
 | |
|     {
 | |
|         return $this->modifiers->count() > 0;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param string $query
 | |
|      */
 | |
|     public function parseQuery(string $query): void
 | |
|     {
 | |
|         $filteredQuery       = app('steam')->cleanString($query);
 | |
|         $this->originalQuery = $filteredQuery;
 | |
|         $pattern             = '/[[:alpha:]_]*:(".*"|[\P{Zs}_-]*)/ui';
 | |
|         $matches             = [];
 | |
|         preg_match_all($pattern, $filteredQuery, $matches);
 | |
| 
 | |
|         foreach ($matches[0] as $match) {
 | |
|             $this->extractModifier($match);
 | |
|             $filteredQuery = str_replace($match, '', $filteredQuery);
 | |
|         }
 | |
|         $filteredQuery = trim(str_replace(['"', "'"], '', $filteredQuery));
 | |
| 
 | |
|         // str replace some stuff:
 | |
|         $search        = ['%', '=', '/', '<', '>', '(', ')', ';'];
 | |
|         $filteredQuery = str_replace($search, ' ', $filteredQuery);
 | |
| 
 | |
|         if ('' !== $filteredQuery) {
 | |
|             $this->words = array_map('trim', explode(' ', $filteredQuery));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return float
 | |
|      */
 | |
|     public function searchTime(): float
 | |
|     {
 | |
|         return microtime(true) - $this->startTime;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return LengthAwarePaginator
 | |
|      */
 | |
|     public function searchTransactions(): LengthAwarePaginator
 | |
|     {
 | |
|         Log::debug('Start of searchTransactions()');
 | |
|         $pageSize = (int) config('firefly.search_result_limit');
 | |
| 
 | |
|         /** @var GroupCollectorInterface $collector */
 | |
|         $collector = app(GroupCollectorInterface::class);
 | |
| 
 | |
|         $collector->setLimit($pageSize)->setPage($this->page)->withAccountInformation();
 | |
|         $collector->withCategoryInformation()->withBudgetInformation();
 | |
|         $collector->setSearchWords($this->words);
 | |
| 
 | |
|         // Most modifiers can be applied to the collector directly.
 | |
|         $collector = $this->applyModifiers($collector);
 | |
| 
 | |
|         return $collector->getPaginatedGroups();
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param User $user
 | |
|      */
 | |
|     public function setUser(User $user): void
 | |
|     {
 | |
|         $this->user = $user;
 | |
|         $this->accountRepository->setUser($user);
 | |
|         $this->billRepository->setUser($user);
 | |
|         $this->categoryRepository->setUser($user);
 | |
|         $this->budgetRepository->setUser($user);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param GroupCollectorInterface $collector
 | |
|      *
 | |
|      * @return GroupCollectorInterface
 | |
|      *
 | |
|      */
 | |
|     private function applyModifiers(GroupCollectorInterface $collector): GroupCollectorInterface
 | |
|     {
 | |
|         /*
 | |
|          * TODO:
 | |
|          * 'bill'?
 | |
|          */
 | |
|         $totalAccounts = new Collection;
 | |
| 
 | |
|         foreach ($this->modifiers as $modifier) {
 | |
|             switch ($modifier['type']) {
 | |
|                 default:
 | |
|                     die(sprintf('unsupported modifier: "%s"', $modifier['type']));
 | |
|                 case 'from':
 | |
|                 case 'source':
 | |
|                     // source can only be asset, liability or revenue account:
 | |
|                     $searchTypes = [AccountType::ASSET, AccountType::MORTGAGE, AccountType::LOAN, AccountType::DEBT, AccountType::REVENUE];
 | |
|                     $accounts    = $this->accountRepository->searchAccount($modifier['value'], $searchTypes, 25);
 | |
|                     if ($accounts->count() > 0) {
 | |
|                         $totalAccounts = $accounts->merge($totalAccounts);
 | |
|                     }
 | |
|                     break;
 | |
|                 case 'to':
 | |
|                 case 'destination':
 | |
|                     // source can only be asset, liability or expense account:
 | |
|                     $searchTypes = [AccountType::ASSET, AccountType::MORTGAGE, AccountType::LOAN, AccountType::DEBT, AccountType::EXPENSE];
 | |
|                     $accounts    = $this->accountRepository->searchAccount($modifier['value'], $searchTypes, 25);
 | |
|                     if ($accounts->count() > 0) {
 | |
|                         $totalAccounts = $accounts->merge($totalAccounts);
 | |
|                     }
 | |
|                     break;
 | |
|                 case 'category':
 | |
|                     $result = $this->categoryRepository->searchCategory($modifier['value'], 25);
 | |
|                     if ($result->count() > 0) {
 | |
|                         $collector->setCategories($result);
 | |
|                     }
 | |
|                     break;
 | |
|                 case 'bill':
 | |
|                     $result = $this->billRepository->searchBill($modifier['value'], 25);
 | |
|                     if ($result->count() > 0) {
 | |
|                         $collector->setBills($result);
 | |
|                     }
 | |
|                     break;
 | |
|                 case 'tag':
 | |
|                     $result = $this->tagRepository->searchTag($modifier['value']);
 | |
|                     if ($result->count() > 0) {
 | |
|                         $collector->setTags($result);
 | |
|                     }
 | |
|                     break;
 | |
|                     break;
 | |
|                 case 'budget':
 | |
|                     $result = $this->budgetRepository->searchBudget($modifier['value'], 25);
 | |
|                     if ($result->count() > 0) {
 | |
|                         $collector->setBudgets($result);
 | |
|                     }
 | |
|                     break;
 | |
|                 case 'amount_is':
 | |
|                 case 'amount':
 | |
|                     $amount = app('steam')->positive((string)$modifier['value']);
 | |
|                     Log::debug(sprintf('Set "%s" using collector with value "%s"', $modifier['type'], $amount));
 | |
|                     $collector->amountIs($amount);
 | |
|                     break;
 | |
|                 case 'amount_max':
 | |
|                 case 'amount_less':
 | |
|                     $amount = app('steam')->positive((string)$modifier['value']);
 | |
|                     Log::debug(sprintf('Set "%s" using collector with value "%s"', $modifier['type'], $amount));
 | |
|                     $collector->amountLess($amount);
 | |
|                     break;
 | |
|                 case 'amount_min':
 | |
|                 case 'amount_more':
 | |
|                     $amount = app('steam')->positive((string)$modifier['value']);
 | |
|                     Log::debug(sprintf('Set "%s" using collector with value "%s"', $modifier['type'], $amount));
 | |
|                     $collector->amountMore($amount);
 | |
|                     break;
 | |
|                 case 'type':
 | |
|                     $collector->setTypes([ucfirst($modifier['value'])]);
 | |
|                     Log::debug(sprintf('Set "%s" using collector with value "%s"', $modifier['type'], $modifier['value']));
 | |
|                     break;
 | |
|                 case 'date':
 | |
|                 case 'on':
 | |
|                     Log::debug(sprintf('Set "%s" using collector with value "%s"', $modifier['type'], $modifier['value']));
 | |
|                     $start = new Carbon($modifier['value']);
 | |
|                     $collector->setRange($start, $start);
 | |
|                     break;
 | |
|                 case 'date_before':
 | |
|                 case 'before':
 | |
|                     Log::debug(sprintf('Set "%s" using collector with value "%s"', $modifier['type'], $modifier['value']));
 | |
|                     $before = new Carbon($modifier['value']);
 | |
|                     $collector->setBefore($before);
 | |
|                     break;
 | |
|                 case 'date_after':
 | |
|                 case 'after':
 | |
|                     Log::debug(sprintf('Set "%s" using collector with value "%s"', $modifier['type'], $modifier['value']));
 | |
|                     $after = new Carbon($modifier['value']);
 | |
|                     $collector->setAfter($after);
 | |
|                     break;
 | |
|                 case 'created_on':
 | |
|                     Log::debug(sprintf('Set "%s" using collector with value "%s"', $modifier['type'], $modifier['value']));
 | |
|                     $createdAt = new Carbon($modifier['value']);
 | |
|                     $collector->setCreatedAt($createdAt);
 | |
|                     break;
 | |
|                 case 'updated_on':
 | |
|                     Log::debug(sprintf('Set "%s" using collector with value "%s"', $modifier['type'], $modifier['value']));
 | |
|                     $updatedAt = new Carbon($modifier['value']);
 | |
|                     $collector->setUpdatedAt($updatedAt);
 | |
|                     break;
 | |
|                 case 'external_id':
 | |
|                     $collector->setExternalId($modifier['value']);
 | |
|                     break;
 | |
|                 case 'internal_reference':
 | |
|                     $collector->setInternalReference($modifier['value']);
 | |
|                     break;
 | |
|             }
 | |
|         }
 | |
|         $collector->setAccounts($totalAccounts);
 | |
| 
 | |
|         return $collector;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param string $string
 | |
|      */
 | |
|     private function extractModifier(string $string): void
 | |
|     {
 | |
|         $parts = explode(':', $string);
 | |
|         if (2 === count($parts) && '' !== trim((string)$parts[1]) && '' !== trim((string)$parts[0])) {
 | |
|             $type  = strtolower(trim((string)$parts[0]));
 | |
|             $value = trim((string)$parts[1]);
 | |
|             $value = trim(trim($value, '"\''));
 | |
|             if (in_array($type, $this->validModifiers, true)) {
 | |
|                 // filter for valid type
 | |
|                 $this->modifiers->push(['type' => $type, 'value' => $value]);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param int $page
 | |
|      */
 | |
|     public function setPage(int $page): void
 | |
|     {
 | |
|         $this->page = $page;
 | |
|     }
 | |
| }
 |