mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-22 20:16:22 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			160 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			160 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * UniqueIban.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\Rules;
 | |
| 
 | |
| use FireflyIII\Models\Account;
 | |
| use FireflyIII\Models\AccountType;
 | |
| use Illuminate\Contracts\Validation\Rule;
 | |
| use Log;
 | |
| 
 | |
| /**
 | |
|  * Class UniqueIban
 | |
|  */
 | |
| class UniqueIban implements Rule
 | |
| {
 | |
|     private ?Account $account;
 | |
|     private ?string  $expectedType;
 | |
| 
 | |
|     /**
 | |
|      * Create a new rule instance.
 | |
|      *
 | |
|      * @codeCoverageIgnore
 | |
|      *
 | |
|      * @param Account|null $account
 | |
|      * @param string|null  $expectedType
 | |
|      */
 | |
|     public function __construct(?Account $account, ?string $expectedType)
 | |
|     {
 | |
|         $this->account      = $account;
 | |
|         $this->expectedType = $expectedType;
 | |
|         // a very basic fix to make sure we get the correct account type:
 | |
|         if ('expense' === $expectedType) {
 | |
|             $this->expectedType = AccountType::EXPENSE;
 | |
|         }
 | |
|         if ('revenue' === $expectedType) {
 | |
|             $this->expectedType = AccountType::REVENUE;
 | |
|         }
 | |
|         if ('asset' === $expectedType) {
 | |
|             $this->expectedType = AccountType::ASSET;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the validation error message.
 | |
|      *
 | |
|      * @codeCoverageIgnore
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     public function message(): string
 | |
|     {
 | |
|         return (string)trans('validation.unique_iban_for_user');
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Determine if the validation rule passes.
 | |
|      *
 | |
|      * @param string $attribute
 | |
|      * @param mixed  $value
 | |
|      *
 | |
|      * @return bool
 | |
|      *
 | |
|      */
 | |
|     public function passes($attribute, $value): bool
 | |
|     {
 | |
|         if (!auth()->check()) {
 | |
|             return true;
 | |
|         }
 | |
|         if (null === $this->expectedType) {
 | |
|             return true;
 | |
|         }
 | |
|         $maxCounts = $this->getMaxOccurrences();
 | |
| 
 | |
|         foreach ($maxCounts as $type => $max) {
 | |
|             $count = $this->countHits($type, $value);
 | |
|             Log::debug(sprintf('Count for "%s" and IBAN "%s" is %d', $type, $value, $count));
 | |
|             if ($count > $max) {
 | |
|                 Log::debug(
 | |
|                     sprintf(
 | |
|                         'IBAN "%s" is in use with %d account(s) of type "%s", which is too much for expected type "%s"',
 | |
|                         $value, $count, $type, $this->expectedType
 | |
|                     )
 | |
|                 );
 | |
| 
 | |
|                 return false;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return array
 | |
|      *
 | |
|      */
 | |
|     private function getMaxOccurrences(): array
 | |
|     {
 | |
|         $maxCounts = [
 | |
|             AccountType::ASSET   => 0,
 | |
|             AccountType::EXPENSE => 0,
 | |
|             AccountType::REVENUE => 0,
 | |
|         ];
 | |
| 
 | |
|         if ('expense' === $this->expectedType || AccountType::EXPENSE === $this->expectedType) {
 | |
|             // IBAN should be unique amongst expense and asset accounts.
 | |
|             // may appear once in revenue accounts
 | |
|             $maxCounts[AccountType::REVENUE] = 1;
 | |
|         }
 | |
|         if ('revenue' === $this->expectedType || AccountType::REVENUE === $this->expectedType) {
 | |
|             // IBAN should be unique amongst revenue and asset accounts.
 | |
|             // may appear once in expense accounts
 | |
|             $maxCounts[AccountType::EXPENSE] = 1;
 | |
|         }
 | |
| 
 | |
|         return $maxCounts;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param string $type
 | |
|      * @param string $iban
 | |
|      *
 | |
|      * @return int
 | |
|      */
 | |
|     private function countHits(string $type, string $iban): int
 | |
|     {
 | |
|         $query
 | |
|             = auth()->user()
 | |
|                     ->accounts()
 | |
|                     ->leftJoin('account_types', 'account_types.id', '=', 'accounts.account_type_id')
 | |
|                     ->where('accounts.iban', $iban)
 | |
|                     ->where('account_types.type', $type);
 | |
| 
 | |
|         if (null !== $this->account) {
 | |
|             $query->where('accounts.id', '!=', $this->account->id);
 | |
|         }
 | |
| 
 | |
|         return $query->count();
 | |
|     }
 | |
| }
 |