. */ declare(strict_types=1); namespace FireflyIII\Factory; use FireflyIII\Exceptions\FireflyException; use FireflyIII\Models\PiggyBank; use FireflyIII\Models\TransactionCurrency; use FireflyIII\Repositories\Account\AccountRepositoryInterface; use FireflyIII\Repositories\Currency\CurrencyRepositoryInterface; use FireflyIII\Repositories\PiggyBank\PiggyBankRepositoryInterface; use FireflyIII\User; use Illuminate\Database\QueryException; /** * Class PiggyBankFactory */ class PiggyBankFactory { public User $user { set(User $value) { $this->user = $value; } } private CurrencyRepositoryInterface $currencyRepository; private AccountRepositoryInterface $accountRepository; private PiggyBankRepositoryInterface $piggyBankRepository; /** * Store a piggy bank or come back with an exception. * * @param array $data * * @return PiggyBank */ public function store(array $data): PiggyBank { $this->currencyRepository = app(CurrencyRepositoryInterface::class); $this->accountRepository = app(AccountRepositoryInterface::class); $this->piggyBankRepository = app(PiggyBankRepositoryInterface::class); $this->currencyRepository->setUser($this->user); $this->accountRepository->setUser($this->user); $this->piggyBankRepository->setUser($this->user); $piggyBankData =$data; // unset some fields unset($piggyBankData['object_group_title'],$piggyBankData['transaction_currency_code'],$piggyBankData['transaction_currency_id'],$piggyBankData['accounts'], $piggyBankData['object_group_id'], $piggyBankData['notes']); // validate amount: if (array_key_exists('target_amount', $piggyBankData) && '' === (string)$piggyBankData['target_amount']) { $piggyBankData['target_amount'] = '0'; } $piggyBankData['start_date_tz'] = $piggyBankData['start_date']?->format('e'); $piggyBankData['target_date_tz'] = $piggyBankData['target_date']?->format('e'); $piggyBankData['account_id'] = null; $piggyBankData['transaction_currency_id'] = $this->getCurrency($data)->id; $piggyBankData['order'] = 131337; try { /** @var PiggyBank $piggyBank */ $piggyBank = PiggyBank::create($piggyBankData); } catch (QueryException $e) { app('log')->error(sprintf('Could not store piggy bank: %s', $e->getMessage()), $piggyBankData); throw new FireflyException('400005: Could not store new piggy bank.', 0, $e); } $piggyBank = $this->setOrder($piggyBank, $data); $this->linkToAccountIds($piggyBank, $data['accounts']); $this->piggyBankRepository->updateNote($piggyBank, $data['notes']); $objectGroupTitle = $data['object_group_title'] ?? ''; if ('' !== $objectGroupTitle) { $objectGroup = $this->findOrCreateObjectGroup($objectGroupTitle); if (null !== $objectGroup) { $piggyBank->objectGroups()->sync([$objectGroup->id]); $piggyBank->save(); } } // try also with ID $objectGroupId = (int)($data['object_group_id'] ?? 0); if (0 !== $objectGroupId) { $objectGroup = $this->findObjectGroupById($objectGroupId); if (null !== $objectGroup) { $piggyBank->objectGroups()->sync([$objectGroup->id]); $piggyBank->save(); } } return $piggyBank; } public function find(?int $piggyBankId, ?string $piggyBankName): ?PiggyBank { $piggyBankId = (int)$piggyBankId; $piggyBankName = (string)$piggyBankName; if ('' === $piggyBankName && 0 === $piggyBankId) { return null; } // first find by ID: if ($piggyBankId > 0) { /** @var null|PiggyBank $piggyBank */ $piggyBank = $this->user->piggyBanks()->find($piggyBankId); if (null !== $piggyBank) { return $piggyBank; } } // then find by name: if ('' !== $piggyBankName) { /** @var null|PiggyBank $piggyBank */ $piggyBank = $this->findByName($piggyBankName); if (null !== $piggyBank) { return $piggyBank; } } return null; } public function findByName(string $name): ?PiggyBank { return $this->user->piggyBanks()->where('piggy_banks.name', $name)->first(); } private function getCurrency(array $data): TransactionCurrency { // currency: $defaultCurrency = app('amount')->getDefaultCurrency(); $currency = null; if (array_key_exists('transaction_currency_code', $data)) { $currency = $this->currencyRepository->findByCode((string)($data['transaction_currency_code'] ?? '')); } if (array_key_exists('transaction_currency_id', $data)) { $currency = $this->currencyRepository->find((int)($data['transaction_currency_id'] ?? 0)); } $currency ??= $defaultCurrency; return $currency; } private function setOrder(PiggyBank $piggyBank, array $data): PiggyBank { $this->resetOrder(); $order = $this->getMaxOrder() + 1; if (array_key_exists('order', $data)) { $order = $data['order']; } $piggyBank->order = $order; $piggyBank->save(); return $piggyBank; } public function resetOrder(): void { // TODO duplicate code $set = PiggyBank ::leftJoin('account_piggy_bank', 'account_piggy_bank.piggy_bank_id', '=', 'piggy_banks.id') ->leftJoin('accounts', 'accounts.id', '=', 'account_piggy_bank.account_id') ->where('accounts.user_id', $this->user->id) ->with( [ 'account', 'objectGroups', ] ) ->orderBy('piggy_banks.order', 'ASC')->get(['piggy_banks.*']); $current = 1; foreach ($set as $piggyBank) { if ($piggyBank->order !== $current) { app('log')->debug(sprintf('Piggy bank #%d ("%s") was at place %d but should be on %d', $piggyBank->id, $piggyBank->name, $piggyBank->order, $current)); $piggyBank->order = $current; $piggyBank->save(); } ++$current; } } private function getMaxOrder(): int { return (int) $this->piggyBankRepository->getPiggyBanks()->max('order'); } private function linkToAccountIds(PiggyBank $piggyBank, array $accounts): void { /** @var array $info */ foreach($accounts as $info) { $account = $this->accountRepository->find((int)($info['account_id'] ?? 0)); if(null === $account) { continue; } $piggyBank->accounts()->syncWithoutDetaching([$account->id => ['current_amount' => $info['current_amount'] ?? '0']]); } } }