. */ declare(strict_types=1); namespace FireflyIII\Rules; use FireflyIII\Models\Account; use FireflyIII\Models\AccountType; use Illuminate\Contracts\Validation\Rule; use Illuminate\Support\Collection; use Log; /** * Class UniqueIban */ class UniqueIban implements Rule { /** @var Account */ private $account; /** @var string */ private $expectedType; /** * Create a new rule instance. * * @param Account|null $account * @param string|null $expectedType */ public function __construct(?Account $account, ?string $expectedType) { $this->account = $account; $this->expectedType = $expectedType; } /** * Get the validation error message. * * @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 * @SuppressWarnings(PHPMD.CyclomaticComplexity) * @SuppressWarnings(PHPMD.UnusedFormalParameter) */ public function passes($attribute, $value): bool { if (!auth()->check()) { return true; // @codeCoverageIgnore } if (null === $this->expectedType) { return true; } $maxCounts = $this->getMaxOccurrences(); foreach ($maxCounts as $type => $max) { $count = $this->countHits($type, $value); 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; } /** * @param string $type * @param string $iban * * @return int */ private function countHits(string $type, string $iban): int { $count = 0; /** @noinspection NullPointerExceptionInspection */ $query = auth()->user() ->accounts() ->leftJoin('account_types', 'account_types.id', '=', 'accounts.account_type_id') ->where('account_types.type', $type); if (null !== $this->account) { $query->where('accounts.id', '!=', $this->account->id); } /** @var Collection $result */ $result = $query->get(['accounts.*']); foreach ($result as $account) { if ($account->iban === $iban) { $count++; } } return $count; } /** * @return array * @SuppressWarnings(PHPMD.CyclomaticComplexity) */ 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::EXPENSE === $this->expectedType) { // IBAN should be unique amongst revenue and asset accounts. // may appear once in expense accounts $maxCounts[AccountType::EXPENSE] = 1; } return $maxCounts; } }