. */ 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\ObjectGroup\CreatesObjectGroups; use FireflyIII\Repositories\PiggyBank\PiggyBankRepositoryInterface; use FireflyIII\User; use Illuminate\Database\QueryException; use Illuminate\Support\Facades\Log; /** * Class PiggyBankFactory */ class PiggyBankFactory { use CreatesObjectGroups; public User $user; private AccountRepositoryInterface $accountRepository; private CurrencyRepositoryInterface $currencyRepository; private PiggyBankRepositoryInterface $piggyBankRepository; public function __construct() { $this->currencyRepository = app(CurrencyRepositoryInterface::class); $this->accountRepository = app(AccountRepositoryInterface::class); $this->piggyBankRepository = app(PiggyBankRepositoryInterface::class); } public function setUser(User $user): void { $this->user = $user; $this->currencyRepository->setUser($user); $this->accountRepository->setUser($user); $this->piggyBankRepository->setUser($user); } /** * Store a piggy bank or come back with an exception. */ public function store(array $data): PiggyBank { $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::createQuietly($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]); } } // 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]); } } Log::debug('Touch piggy bank'); $piggyBank->encrypted = false; $piggyBank->save(); $piggyBank->touch(); return $piggyBank; } private function getCurrency(array $data): TransactionCurrency { // currency: $defaultCurrency = app('amount')->getNativeCurrency(); $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; } 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) { $piggyBank = 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) ->where('piggy_banks.id', $piggyBankId) ->first(['piggy_banks.*']) ; 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 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) ->where('piggy_banks.name', $name) ->first(['piggy_banks.*']) ; } 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->saveQuietly(); 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( [ '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'); } public function linkToAccountIds(PiggyBank $piggyBank, array $accounts): void { Log::debug(sprintf('Linking piggy bank #%d to %d accounts.', $piggyBank->id, count($accounts)), $accounts); // collect current current_amount so the sync does not remove them. // TODO this is a tedious check. Feels like a hack. $toBeLinked = []; foreach ($piggyBank->accounts as $account) { foreach ($accounts as $info) { if ($account->id === $info['account_id']) { if (array_key_exists($account->id, $accounts)) { $toBeLinked[$account->id] = ['current_amount' => $account->pivot->current_amount ?? '0']; Log::debug(sprintf('Prefilled for account #%d with amount %s', $account->id, $account->pivot->current_amount ?? '0')); } } } } /** @var array $info */ foreach ($accounts as $info) { $account = $this->accountRepository->find((int) ($info['account_id'] ?? 0)); if (null === $account) { continue; } if (array_key_exists('current_amount', $info)) { $toBeLinked[$account->id] = ['current_amount' => $info['current_amount']]; Log::debug(sprintf('Will link account #%d with amount %s', $account->id, $account->pivot->current_amount ?? '0')); } if (!array_key_exists('current_amount', $info)) { $toBeLinked[$account->id] ??= []; Log::debug(sprintf('Will link account #%d with info: ', $account->id), $toBeLinked[$account->id]); } } Log::debug(sprintf('Link information: %s', json_encode($toBeLinked))); $piggyBank->accounts()->sync($toBeLinked); } }