| 
									
										
										
										
											2015-02-09 07:23:39 +01:00
										 |  |  | <?php | 
					
						
							| 
									
										
										
										
											2016-05-20 12:41:23 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * JournalRepository.php | 
					
						
							| 
									
										
										
										
											2017-10-21 08:40:00 +02:00
										 |  |  |  * Copyright (c) 2017 thegrumpydictator@gmail.com | 
					
						
							| 
									
										
										
										
											2016-05-20 12:41:23 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2017-10-21 08:40:00 +02:00
										 |  |  |  * This file is part of Firefly III. | 
					
						
							| 
									
										
										
										
											2016-10-05 06:52:15 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2017-10-21 08:40:00 +02:00
										 |  |  |  * Firefly III is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  |  * the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  |  * (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Firefly III 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 General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with Firefly III.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2016-05-20 12:41:23 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-03-24 15:01:53 +01:00
										 |  |  | declare(strict_types=1); | 
					
						
							| 
									
										
										
										
											2015-02-09 07:23:39 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Repositories\Journal; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-24 22:53:38 +01:00
										 |  |  | use FireflyIII\Models\Account; | 
					
						
							| 
									
										
										
										
											2017-11-05 19:48:43 +01:00
										 |  |  | use FireflyIII\Models\Transaction; | 
					
						
							| 
									
										
										
										
											2015-02-24 22:53:38 +01:00
										 |  |  | use FireflyIII\Models\TransactionJournal; | 
					
						
							|  |  |  | use FireflyIII\Models\TransactionType; | 
					
						
							| 
									
										
										
										
											2016-03-03 08:55:43 +01:00
										 |  |  | use FireflyIII\User; | 
					
						
							| 
									
										
										
										
											2016-10-21 21:41:31 +02:00
										 |  |  | use Illuminate\Support\Collection; | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  | use Illuminate\Support\MessageBag; | 
					
						
							| 
									
										
										
										
											2015-04-03 22:54:21 +02:00
										 |  |  | use Log; | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  | use Preferences; | 
					
						
							| 
									
										
										
										
											2015-02-24 22:53:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-11 07:35:10 +01:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |  * Class JournalRepository. | 
					
						
							| 
									
										
										
										
											2015-02-11 07:35:10 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | class JournalRepository implements JournalRepositoryInterface | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |     use CreateJournalsTrait, UpdateJournalsTrait, SupportJournalsTrait; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-03 08:55:43 +01:00
										 |  |  |     /** @var User */ | 
					
						
							|  |  |  |     private $user; | 
					
						
							| 
									
										
										
										
											2016-09-09 11:19:40 +02:00
										 |  |  |     /** @var array */ | 
					
						
							| 
									
										
										
										
											2017-10-03 10:30:56 +02:00
										 |  |  |     private $validMetaFields = ['interest_date', 'book_date', 'process_date', 'due_date', 'payment_date', 'invoice_date', 'internal_reference']; | 
					
						
							| 
									
										
										
										
											2016-09-09 11:19:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param TransactionJournal $journal | 
					
						
							|  |  |  |      * @param TransactionType    $type | 
					
						
							|  |  |  |      * @param Account            $source | 
					
						
							|  |  |  |      * @param Account            $destination | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return MessageBag | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function convert(TransactionJournal $journal, TransactionType $type, Account $source, Account $destination): MessageBag | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // default message bag that shows errors for everything.
 | 
					
						
							|  |  |  |         $messages = new MessageBag; | 
					
						
							|  |  |  |         $messages->add('source_account_revenue', trans('firefly.invalid_convert_selection')); | 
					
						
							|  |  |  |         $messages->add('destination_account_asset', trans('firefly.invalid_convert_selection')); | 
					
						
							|  |  |  |         $messages->add('destination_account_expense', trans('firefly.invalid_convert_selection')); | 
					
						
							|  |  |  |         $messages->add('source_account_asset', trans('firefly.invalid_convert_selection')); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |         if ($source->id === $destination->id || null === $source->id || null === $destination->id) { | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |             return $messages; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $sourceTransaction             = $journal->transactions()->where('amount', '<', 0)->first(); | 
					
						
							|  |  |  |         $destinationTransaction        = $journal->transactions()->where('amount', '>', 0)->first(); | 
					
						
							|  |  |  |         $sourceTransaction->account_id = $source->id; | 
					
						
							|  |  |  |         $sourceTransaction->save(); | 
					
						
							|  |  |  |         $destinationTransaction->account_id = $destination->id; | 
					
						
							|  |  |  |         $destinationTransaction->save(); | 
					
						
							|  |  |  |         $journal->transaction_type_id = $type->id; | 
					
						
							|  |  |  |         $journal->save(); | 
					
						
							| 
									
										
										
										
											2016-12-18 10:37:59 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // if journal is a transfer now, remove budget:
 | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |         if (TransactionType::TRANSFER === $type->type) { | 
					
						
							| 
									
										
										
										
											2016-12-18 10:37:59 +01:00
										 |  |  |             $journal->budgets()->detach(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |         Preferences::mark(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-18 10:37:59 +01:00
										 |  |  |         return new MessageBag; | 
					
						
							| 
									
										
										
										
											2016-10-30 06:14:07 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-04 16:31:16 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param TransactionJournal $journal | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return int | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function countTransactions(TransactionJournal $journal): int | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $journal->transactions()->count(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-04 23:46:14 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param TransactionJournal $journal | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-04-06 09:27:45 +02:00
										 |  |  |     public function delete(TransactionJournal $journal): bool | 
					
						
							| 
									
										
										
										
											2015-05-04 23:46:14 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $journal->delete(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-23 09:33:54 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param int $journalId | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return TransactionJournal | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-12-04 18:02:19 +01:00
										 |  |  |     public function find(int $journalId): TransactionJournal | 
					
						
							| 
									
										
										
										
											2016-04-23 09:33:54 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-08-26 09:30:52 +02:00
										 |  |  |         $journal = $this->user->transactionJournals()->where('id', $journalId)->first(); | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |         if (null === $journal) { | 
					
						
							| 
									
										
										
										
											2016-04-23 09:33:54 +02:00
										 |  |  |             return new TransactionJournal; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $journal; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-05 19:48:43 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Transaction $transaction | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Transaction|null | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function findOpposingTransaction(Transaction $transaction): ?Transaction | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $opposing = Transaction::leftJoin('transaction_journals', 'transaction_journals.id', '=', 'transactions.transaction_journal_id') | 
					
						
							|  |  |  |                                ->where('transaction_journals.user_id', $this->user->id) | 
					
						
							|  |  |  |                                ->where('transactions.transaction_journal_id', $transaction->transaction_journal_id) | 
					
						
							|  |  |  |                                ->where('transactions.identifier', $transaction->identifier) | 
					
						
							|  |  |  |                                ->where('amount', bcmul($transaction->amount, '-1')) | 
					
						
							|  |  |  |                                ->first(['transactions.*']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $opposing; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * @param int $transactionid | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return Transaction|null | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function findTransaction(int $transactionid): ?Transaction | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $transaction = Transaction::leftJoin('transaction_journals', 'transaction_journals.id', '=', 'transactions.transaction_journal_id') | 
					
						
							|  |  |  |                                   ->where('transaction_journals.user_id', $this->user->id) | 
					
						
							|  |  |  |                                   ->where('transactions.id', $transactionid) | 
					
						
							|  |  |  |                                   ->first(['transactions.*']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $transaction; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-03 22:54:21 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |      * Get users first transaction journal. | 
					
						
							| 
									
										
										
										
											2015-04-03 22:54:21 +02:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @return TransactionJournal | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-04-06 09:27:45 +02:00
										 |  |  |     public function first(): TransactionJournal | 
					
						
							| 
									
										
										
										
											2015-04-03 22:54:21 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-08-26 09:30:52 +02:00
										 |  |  |         $entry = $this->user->transactionJournals()->orderBy('date', 'ASC')->first(['transaction_journals.*']); | 
					
						
							| 
									
										
										
										
											2016-05-05 21:25:20 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |         if (null === $entry) { | 
					
						
							| 
									
										
										
										
											2016-04-10 21:31:00 +02:00
										 |  |  |             return new TransactionJournal; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-12-25 17:11:55 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return $entry; | 
					
						
							| 
									
										
										
										
											2015-04-03 22:54:21 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-20 19:50:22 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @return Collection | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function getTransactionTypes(): Collection | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return TransactionType::orderBy('type', 'ASC')->get(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-28 07:51:09 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param TransactionJournal $journal | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function isTransfer(TransactionJournal $journal): bool | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |         return TransactionType::TRANSFER === $journal->transactionType->type; | 
					
						
							| 
									
										
										
										
											2017-04-28 07:51:09 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-05 19:48:43 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param Transaction $transaction | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function reconcile(Transaction $transaction): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         Log::debug(sprintf('Going to reconcile transaction #%d', $transaction->id)); | 
					
						
							|  |  |  |         $opposing = $this->findOpposingTransaction($transaction); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |         if (null === $opposing) { | 
					
						
							| 
									
										
										
										
											2017-11-05 19:48:43 +01:00
										 |  |  |             Log::debug('Opposing transaction is NULL. Cannot reconcile.'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         Log::debug(sprintf('Opposing transaction ID is #%d', $opposing->id)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $transaction->reconciled = true; | 
					
						
							|  |  |  |         $opposing->reconciled    = true; | 
					
						
							|  |  |  |         $transaction->save(); | 
					
						
							|  |  |  |         $opposing->save(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-24 15:01:53 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param TransactionJournal $journal | 
					
						
							|  |  |  |      * @param int                $order | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function setOrder(TransactionJournal $journal, int $order): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $journal->order = $order; | 
					
						
							|  |  |  |         $journal->save(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-17 06:42:36 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param User $user | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function setUser(User $user) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->user = $user; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-24 22:53:38 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * @param array $data | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return TransactionJournal | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-04-06 09:27:45 +02:00
										 |  |  |     public function store(array $data): TransactionJournal | 
					
						
							| 
									
										
										
										
											2015-02-24 22:53:38 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         // find transaction type.
 | 
					
						
							| 
									
										
										
										
											2017-04-14 11:19:09 +02:00
										 |  |  |         /** @var TransactionType $transactionType */ | 
					
						
							| 
									
										
										
										
											2015-02-24 22:53:38 +01:00
										 |  |  |         $transactionType = TransactionType::where('type', ucfirst($data['what']))->first(); | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |         $accounts        = $this->storeAccounts($this->user, $transactionType, $data); | 
					
						
							|  |  |  |         $data            = $this->verifyNativeAmount($data, $accounts); | 
					
						
							| 
									
										
										
										
											2017-04-14 11:19:09 +02:00
										 |  |  |         $amount          = strval($data['amount']); | 
					
						
							| 
									
										
										
										
											2017-04-14 15:42:54 +02:00
										 |  |  |         $journal         = new TransactionJournal( | 
					
						
							| 
									
										
										
										
											2015-02-24 22:53:38 +01:00
										 |  |  |             [ | 
					
						
							| 
									
										
										
										
											2016-10-23 12:19:32 +02:00
										 |  |  |                 'user_id'                 => $this->user->id, | 
					
						
							| 
									
										
										
										
											2015-12-28 16:52:21 +01:00
										 |  |  |                 'transaction_type_id'     => $transactionType->id, | 
					
						
							| 
									
										
										
										
											2017-06-04 13:39:16 +02:00
										 |  |  |                 'transaction_currency_id' => $data['currency_id'], // no longer used.
 | 
					
						
							| 
									
										
										
										
											2015-12-28 16:52:21 +01:00
										 |  |  |                 'description'             => $data['description'], | 
					
						
							|  |  |  |                 'completed'               => 0, | 
					
						
							|  |  |  |                 'date'                    => $data['date'], | 
					
						
							| 
									
										
										
										
											2015-02-24 22:53:38 +01:00
										 |  |  |             ] | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         $journal->save(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-21 19:06:22 +02:00
										 |  |  |         // store stuff:
 | 
					
						
							| 
									
										
										
										
											2017-09-20 08:35:20 +02:00
										 |  |  |         $this->storeCategoryWithJournal($journal, strval($data['category'])); | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |         $this->storeBudgetWithJournal($journal, $data['budget_id']); | 
					
						
							| 
									
										
										
										
											2017-04-14 11:19:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-21 19:06:22 +02:00
										 |  |  |         // store two transactions:
 | 
					
						
							|  |  |  |         $one = [ | 
					
						
							| 
									
										
										
										
											2017-06-04 13:39:16 +02:00
										 |  |  |             'journal'                 => $journal, | 
					
						
							|  |  |  |             'account'                 => $accounts['source'], | 
					
						
							|  |  |  |             'amount'                  => bcmul($amount, '-1'), | 
					
						
							|  |  |  |             'transaction_currency_id' => $data['currency_id'], | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |             'foreign_amount'          => null === $data['foreign_amount'] ? null : bcmul(strval($data['foreign_amount']), '-1'), | 
					
						
							| 
									
										
										
										
											2017-06-04 13:39:16 +02:00
										 |  |  |             'foreign_currency_id'     => $data['foreign_currency_id'], | 
					
						
							|  |  |  |             'description'             => null, | 
					
						
							|  |  |  |             'category'                => null, | 
					
						
							|  |  |  |             'budget'                  => null, | 
					
						
							|  |  |  |             'identifier'              => 0, | 
					
						
							| 
									
										
										
										
											2016-10-21 19:06:22 +02:00
										 |  |  |         ]; | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |         $this->storeTransaction($one); | 
					
						
							| 
									
										
										
										
											2016-10-21 19:06:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $two = [ | 
					
						
							| 
									
										
										
										
											2017-06-04 13:39:16 +02:00
										 |  |  |             'journal'                 => $journal, | 
					
						
							|  |  |  |             'account'                 => $accounts['destination'], | 
					
						
							|  |  |  |             'amount'                  => $amount, | 
					
						
							|  |  |  |             'transaction_currency_id' => $data['currency_id'], | 
					
						
							|  |  |  |             'foreign_amount'          => $data['foreign_amount'], | 
					
						
							|  |  |  |             'foreign_currency_id'     => $data['foreign_currency_id'], | 
					
						
							|  |  |  |             'description'             => null, | 
					
						
							|  |  |  |             'category'                => null, | 
					
						
							|  |  |  |             'budget'                  => null, | 
					
						
							|  |  |  |             'identifier'              => 0, | 
					
						
							| 
									
										
										
										
											2016-10-21 19:06:22 +02:00
										 |  |  |         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |         $this->storeTransaction($two); | 
					
						
							| 
									
										
										
										
											2015-02-24 22:53:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-28 14:08:58 +02:00
										 |  |  |         // store tags
 | 
					
						
							|  |  |  |         if (isset($data['tags']) && is_array($data['tags'])) { | 
					
						
							|  |  |  |             $this->saveTags($journal, $data['tags']); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-03 10:30:56 +02:00
										 |  |  |         // update note:
 | 
					
						
							| 
									
										
										
										
											2017-10-06 06:36:00 +02:00
										 |  |  |         if (isset($data['notes'])) { | 
					
						
							|  |  |  |             $this->updateNote($journal, $data['notes']); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-10-03 10:30:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 11:19:40 +02:00
										 |  |  |         foreach ($data as $key => $value) { | 
					
						
							|  |  |  |             if (in_array($key, $this->validMetaFields)) { | 
					
						
							|  |  |  |                 $journal->setMeta($key, $value); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             Log::debug(sprintf('Could not store meta field "%s" with value "%s" for journal #%d', json_encode($key), json_encode($value), $journal->id)); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-02-24 22:53:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-21 19:06:22 +02:00
										 |  |  |         $journal->completed = 1; | 
					
						
							|  |  |  |         $journal->save(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 11:19:40 +02:00
										 |  |  |         return $journal; | 
					
						
							| 
									
										
										
										
											2015-02-24 22:53:38 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-02-27 14:27:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-28 10:36:13 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |      * @param TransactionJournal $journal | 
					
						
							|  |  |  |      * @param array              $data | 
					
						
							| 
									
										
										
										
											2016-05-01 09:42:08 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |      * @return TransactionJournal | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function update(TransactionJournal $journal, array $data): TransactionJournal | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // update actual journal:
 | 
					
						
							|  |  |  |         $journal->description   = $data['description']; | 
					
						
							|  |  |  |         $journal->date          = $data['date']; | 
					
						
							|  |  |  |         $accounts               = $this->storeAccounts($this->user, $journal->transactionType, $data); | 
					
						
							|  |  |  |         $data                   = $this->verifyNativeAmount($data, $accounts); | 
					
						
							|  |  |  |         $data['amount']         = strval($data['amount']); | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |         $data['foreign_amount'] = null === $data['foreign_amount'] ? null : strval($data['foreign_amount']); | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // unlink all categories, recreate them:
 | 
					
						
							|  |  |  |         $journal->categories()->detach(); | 
					
						
							|  |  |  |         $journal->budgets()->detach(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-20 08:35:20 +02:00
										 |  |  |         $this->storeCategoryWithJournal($journal, strval($data['category'])); | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |         $this->storeBudgetWithJournal($journal, $data['budget_id']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // negative because source loses money.
 | 
					
						
							|  |  |  |         $this->updateSourceTransaction($journal, $accounts['source'], $data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // positive because destination gets money.
 | 
					
						
							|  |  |  |         $this->updateDestinationTransaction($journal, $accounts['destination'], $data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $journal->save(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // update tags:
 | 
					
						
							|  |  |  |         if (isset($data['tags']) && is_array($data['tags'])) { | 
					
						
							|  |  |  |             $this->updateTags($journal, $data['tags']); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-03 10:30:56 +02:00
										 |  |  |         // update note:
 | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |         if (isset($data['notes']) && null !== $data['notes']) { | 
					
						
							| 
									
										
										
										
											2017-10-20 07:51:53 +02:00
										 |  |  |             $this->updateNote($journal, strval($data['notes'])); | 
					
						
							| 
									
										
										
										
											2017-10-06 06:36:00 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-10-03 10:30:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |         // update meta fields:
 | 
					
						
							|  |  |  |         $result = $journal->save(); | 
					
						
							|  |  |  |         if ($result) { | 
					
						
							|  |  |  |             foreach ($data as $key => $value) { | 
					
						
							|  |  |  |                 if (in_array($key, $this->validMetaFields)) { | 
					
						
							|  |  |  |                     $journal->setMeta($key, $value); | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 Log::debug(sprintf('Could not store meta field "%s" with value "%s" for journal #%d', json_encode($key), json_encode($value), $journal->id)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return $journal; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $journal; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Same as above but for transaction journal with multiple transactions. | 
					
						
							| 
									
										
										
										
											2016-05-01 09:42:08 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2015-04-28 10:36:13 +02:00
										 |  |  |      * @param TransactionJournal $journal | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |      * @param array              $data | 
					
						
							| 
									
										
										
										
											2015-04-28 10:36:13 +02:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |      * @return TransactionJournal | 
					
						
							| 
									
										
										
										
											2015-04-28 10:36:13 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |     public function updateSplitJournal(TransactionJournal $journal, array $data): TransactionJournal | 
					
						
							| 
									
										
										
										
											2015-04-28 10:36:13 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |         // update actual journal:
 | 
					
						
							|  |  |  |         $journal->description = $data['journal_description']; | 
					
						
							|  |  |  |         $journal->date        = $data['date']; | 
					
						
							|  |  |  |         $journal->save(); | 
					
						
							|  |  |  |         Log::debug(sprintf('Updated split journal #%d', $journal->id)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // unlink all categories:
 | 
					
						
							|  |  |  |         $journal->categories()->detach(); | 
					
						
							|  |  |  |         $journal->budgets()->detach(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-03 10:30:56 +02:00
										 |  |  |         // update note:
 | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |         if (isset($data['notes']) && null !== $data['notes']) { | 
					
						
							| 
									
										
										
										
											2017-10-20 07:51:53 +02:00
										 |  |  |             $this->updateNote($journal, strval($data['notes'])); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |         // update meta fields:
 | 
					
						
							|  |  |  |         $result = $journal->save(); | 
					
						
							|  |  |  |         if ($result) { | 
					
						
							|  |  |  |             foreach ($data as $key => $value) { | 
					
						
							|  |  |  |                 if (in_array($key, $this->validMetaFields)) { | 
					
						
							|  |  |  |                     $journal->setMeta($key, $value); | 
					
						
							|  |  |  |                     continue; | 
					
						
							| 
									
										
										
										
											2016-05-01 09:42:08 +02:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |                 Log::debug(sprintf('Could not store meta field "%s" with value "%s" for journal #%d', json_encode($key), json_encode($value), $journal->id)); | 
					
						
							| 
									
										
										
										
											2015-04-28 21:01:54 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-04-28 10:36:13 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |         // update tags:
 | 
					
						
							|  |  |  |         if (isset($data['tags']) && is_array($data['tags'])) { | 
					
						
							|  |  |  |             $this->updateTags($journal, $data['tags']); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // delete original transactions, and recreate them.
 | 
					
						
							|  |  |  |         $journal->transactions()->delete(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // store each transaction.
 | 
					
						
							|  |  |  |         $identifier = 0; | 
					
						
							|  |  |  |         Log::debug(sprintf('Count %d transactions in updateSplitJournal()', count($data['transactions']))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach ($data['transactions'] as $transaction) { | 
					
						
							|  |  |  |             Log::debug(sprintf('Split journal update split transaction %d', $identifier)); | 
					
						
							|  |  |  |             $transaction = $this->appendTransactionData($transaction, $data); | 
					
						
							|  |  |  |             $this->storeSplitTransaction($journal, $transaction, $identifier); | 
					
						
							| 
									
										
										
										
											2017-11-15 12:25:49 +01:00
										 |  |  |             ++$identifier; | 
					
						
							| 
									
										
										
										
											2017-06-07 08:18:42 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $journal->save(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $journal; | 
					
						
							| 
									
										
										
										
											2015-04-28 10:36:13 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-04-03 09:30:44 +02:00
										 |  |  | } |