. */ declare(strict_types=1); namespace FireflyIII\Factory; use FireflyIII\Exceptions\FireflyException; use FireflyIII\Models\Account; use FireflyIII\Models\AccountType; use FireflyIII\Models\Transaction; use FireflyIII\Models\TransactionCurrency; use FireflyIII\Models\TransactionJournal; use FireflyIII\Models\TransactionType; use FireflyIII\Repositories\Account\AccountRepositoryInterface; use FireflyIII\Support\NullArrayObject; use FireflyIII\User; use Illuminate\Support\Collection; use Log; /** * Class TransactionFactory */ class TransactionFactory { /** @var AccountRepositoryInterface */ private $accountRepository; /** @var TransactionJournal */ private $journal; /** @var User */ private $user; /** * Constructor. */ public function __construct() { if ('testing' === config('app.env')) { Log::warning(sprintf('%s should not be instantiated in the TEST environment!', \get_class($this))); } $this->accountRepository = app(AccountRepositoryInterface::class); } //use TransactionServiceTrait; /** * @param Account $account * @param TransactionCurrency $currency * @param string $amount * * @return Transaction|null */ public function create(Account $account, TransactionCurrency $currency, string $amount): ?Transaction { $result = Transaction::create( [ 'reconciled' => false, 'account_id' => $account->id, 'transaction_journal_id' => $this->journal->id, 'description' => null, 'transaction_currency_id' => $currency->id, 'amount' => $amount, 'foreign_amount' => null, 'foreign_currency_id' => null, 'identifier' => 0, ] ); if (null !== $result) { Log::debug( sprintf( 'Created transaction #%d (%s %s), part of journal #%d', $result->id, $currency->code, $amount, $this->journal->id ) ); } return $result; } /** * @param NullArrayObject $data * @param TransactionCurrency $currency * @param TransactionCurrency|null $foreignCurrency * * @return Collection * @throws FireflyException */ public function createPair(NullArrayObject $data, TransactionCurrency $currency, ?TransactionCurrency $foreignCurrency): Collection { $sourceAccount = $this->getAccount('source', $data['source'], $data['source_id'], $data['source_name']); $destinationAccount = $this->getAccount('destination', $data['destination'], $data['destination_id'], $data['destination_name']); $amount = $this->getAmount($data['amount']); $foreignAmount = $this->getForeignAmount($data['foreign_amount']); $one = $this->create($sourceAccount, $currency, app('steam')->negative($amount)); $two = $this->create($destinationAccount, $currency, app('steam')->positive($amount)); $one->reconciled = $data['reconciled'] ?? false; $two->reconciled = $data['reconciled'] ?? false; // add foreign currency info to $one and $two if necessary. if (null !== $foreignCurrency) { $one->foreign_currency_id = $foreignCurrency->id; $two->foreign_currency_id = $foreignCurrency->id; $one->foreign_amount = $foreignAmount; $two->foreign_amount = $foreignAmount; } $one->save(); $two->save(); return new Collection([$one, $two]); } /** * @param TransactionJournal $journal */ public function setJournal(TransactionJournal $journal): void { $this->journal = $journal; } /** * @param User $user */ public function setUser(User $user): void { $this->user = $user; $this->accountRepository->setUser($user); } /** * @param string $direction * @param Account|null $source * @param int|null $sourceId * @param string|null $sourceName * * @return Account * @throws FireflyException */ private function getAccount(string $direction, ?Account $source, ?int $sourceId, ?string $sourceName): Account { Log::debug(sprintf('Now in getAccount(%s)', $direction)); Log::debug(sprintf('Parameters: ((account), %s, %s)', var_export($sourceId, true), var_export($sourceName, true))); // expected type of source account, in order of preference $array = [ 'source' => [ TransactionType::WITHDRAWAL => [AccountType::ASSET, AccountType::LOAN, AccountType::DEBT, AccountType::MORTGAGE], TransactionType::DEPOSIT => [AccountType::REVENUE, AccountType::LOAN, AccountType::DEBT, AccountType::MORTGAGE], TransactionType::TRANSFER => [AccountType::ASSET, AccountType::LOAN, AccountType::DEBT, AccountType::MORTGAGE], TransactionType::OPENING_BALANCE => [AccountType::INITIAL_BALANCE, AccountType::ASSET, AccountType::LOAN, AccountType::DEBT, AccountType::MORTGAGE], TransactionType::RECONCILIATION => [AccountType::RECONCILIATION, AccountType::ASSET, AccountType::LOAN, AccountType::DEBT, AccountType::MORTGAGE], ], 'destination' => [ TransactionType::WITHDRAWAL => [AccountType::EXPENSE, AccountType::ASSET], TransactionType::DEPOSIT => [AccountType::ASSET, AccountType::LOAN, AccountType::DEBT, AccountType::MORTGAGE], TransactionType::TRANSFER => [AccountType::ASSET, AccountType::LOAN, AccountType::DEBT, AccountType::MORTGAGE], TransactionType::OPENING_BALANCE => [AccountType::INITIAL_BALANCE, AccountType::ASSET, AccountType::LOAN, AccountType::DEBT, AccountType::MORTGAGE], TransactionType::RECONCILIATION => [AccountType::RECONCILIATION, AccountType::ASSET, AccountType::LOAN, AccountType::DEBT, AccountType::MORTGAGE], ], ]; $expectedTypes = $array[$direction]; unset($array); // and now try to find it, based on the type of transaction. $transactionType = $this->journal->transactionType->type; Log::debug( sprintf( 'Based on the fact that the transaction is a %s, the %s account should be in: %s', $transactionType, $direction, implode(', ', $expectedTypes[$transactionType]) ) ); // first attempt, check the "source" object. if (null !== $source && $source->user_id === $this->user->id && \in_array($source->accountType->type, $expectedTypes[$transactionType], true)) { Log::debug(sprintf('Found "account" object for %s: #%d, %s', $direction, $source->id, $source->name)); return $source; } // second attempt, find by ID. if (null !== $sourceId) { $source = $this->accountRepository->findNull($sourceId); if (null !== $source) { Log::debug(sprintf('Found account #%d ("%s" of type "%s") based on #%d.', $source->id, $source->name, $source->accountType->type, $sourceId)); } if (null !== $source && \in_array($source->accountType->type, $expectedTypes[$transactionType], true)) { Log::debug(sprintf('Found "account_id" object for %s: #%d, %s', $direction, $source->id, $source->name)); return $source; } } // third attempt, find by name. if (null !== $sourceName) { // find by preferred type. $source = $this->accountRepository->findByName($sourceName, [$expectedTypes[$transactionType][0]]); // or any type. $source = $source ?? $this->accountRepository->findByName($sourceName, $expectedTypes[$transactionType]); if (null !== $source) { Log::debug(sprintf('Found "account_name" object for %s: #%d, %s', $direction, $source->id, $source->name)); return $source; } } $sourceName = $sourceName ?? '(no name)'; // final attempt, create it. $preferredType = $expectedTypes[$transactionType][0]; if (AccountType::ASSET === $preferredType) { throw new FireflyException(sprintf('TransactionFactory: Cannot create asset account with ID #%d or name "%s".', $sourceId, $sourceName)); } return $this->accountRepository->store( [ 'account_type_id' => null, 'accountType' => $preferredType, 'name' => $sourceName, 'active' => true, 'iban' => null, ] ); } /** * @param string $amount * * @return string * @throws FireflyException */ private function getAmount(string $amount): string { if ('' === $amount) { throw new FireflyException(sprintf('The amount cannot be an empty string: "%s"', $amount)); } if (0 === bccomp('0', $amount)) { throw new FireflyException(sprintf('The amount seems to be zero: "%s"', $amount)); } return $amount; } /** * @param string|null $amount * * @return string */ private function getForeignAmount(?string $amount): ?string { if (null === $amount) { Log::debug('No foreign amount info in array. Return NULL'); return null; } if ('' === $amount) { Log::debug('Foreign amount is empty string, return NULL.'); return null; } if (0 === bccomp('0', $amount)) { Log::debug('Foreign amount is 0.0, return NULL.'); return null; } Log::debug(sprintf('Foreign amount is %s', $amount)); return $amount; } // // /** // * @param string $sourceType // * @param string $destinationType // * @param string $transactionType // * // * @throws FireflyException // */ // private function validateTransaction(string $sourceType, string $destinationType, string $transactionType): void // { // // throw big fat error when source type === dest type and it's not a transfer or reconciliation. // if ($sourceType === $destinationType && $transactionType !== TransactionType::TRANSFER) { // throw new FireflyException(sprintf('Source and destination account cannot be both of the type "%s"', $destinationType)); // } // // source must be in this list AND dest must be in this list: // $list = [AccountType::DEFAULT, AccountType::ASSET, AccountType::CREDITCARD, AccountType::CASH, AccountType::DEBT, AccountType::MORTGAGE, // AccountType::LOAN, AccountType::MORTGAGE]; // if ( // !\in_array($sourceType, $list, true) // && !\in_array($destinationType, $list, true)) { // throw new FireflyException(sprintf('At least one of the accounts must be an asset account (%s, %s).', $sourceType, $destinationType)); // } // } }