mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-25 05:03:13 +00:00 
			
		
		
		
	
		
			
	
	
		
			385 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			385 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
|   | <?php | ||
|  | /** | ||
|  |  * Importer.php | ||
|  |  * Copyright (C) 2016 thegrumpydictator@gmail.com | ||
|  |  * | ||
|  |  * This software may be modified and distributed under the terms | ||
|  |  * of the MIT license.  See the LICENSE file for details. | ||
|  |  */ | ||
|  | 
 | ||
|  | declare(strict_types = 1); | ||
|  | namespace FireflyIII\Helpers\Csv; | ||
|  | 
 | ||
|  | use Auth; | ||
|  | use FireflyIII\Events\TransactionJournalStored; | ||
|  | use FireflyIII\Exceptions\FireflyException; | ||
|  | use FireflyIII\Helpers\Csv\Converter\ConverterInterface; | ||
|  | use FireflyIII\Helpers\Csv\PostProcessing\PostProcessorInterface; | ||
|  | use FireflyIII\Helpers\Csv\Specifix\SpecifixInterface; | ||
|  | use FireflyIII\Models\Account; | ||
|  | use FireflyIII\Models\Transaction; | ||
|  | use FireflyIII\Models\TransactionJournal; | ||
|  | use FireflyIII\Models\TransactionType; | ||
|  | use Illuminate\Support\Collection; | ||
|  | use Illuminate\Support\MessageBag; | ||
|  | use Log; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Class Importer | ||
|  |  * | ||
|  |  * @package FireflyIII\Helpers\Csv | ||
|  |  */ | ||
|  | class Importer | ||
|  | { | ||
|  | 
 | ||
|  |     /** @var Data */ | ||
|  |     protected $data; | ||
|  |     /** @var array */ | ||
|  |     protected $errors = []; | ||
|  |     /** @var  array */ | ||
|  |     protected $importData; | ||
|  |     /** @var  array */ | ||
|  |     protected $importRow; | ||
|  |     /** @var int */ | ||
|  |     protected $imported = 0; | ||
|  |     /** @var  Collection */ | ||
|  |     protected $journals; | ||
|  |     /** @var array */ | ||
|  |     protected $map; | ||
|  |     /** @var  array */ | ||
|  |     protected $mapped; | ||
|  |     /** @var  array */ | ||
|  |     protected $roles; | ||
|  |     /** @var  int */ | ||
|  |     protected $rows = 0; | ||
|  |     /** @var array */ | ||
|  |     protected $specifix = []; | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Used by CsvController. | ||
|  |      * | ||
|  |      * @return array | ||
|  |      */ | ||
|  |     public function getErrors(): array | ||
|  |     { | ||
|  |         return $this->errors; | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Used by CsvController | ||
|  |      * | ||
|  |      * @return int | ||
|  |      */ | ||
|  |     public function getImported(): int | ||
|  |     { | ||
|  |         return $this->imported; | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @return Collection | ||
|  |      */ | ||
|  |     public function getJournals(): Collection | ||
|  |     { | ||
|  |         return $this->journals; | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Used by CsvController | ||
|  |      * | ||
|  |      * @return int | ||
|  |      */ | ||
|  |     public function getRows(): int | ||
|  |     { | ||
|  |         return $this->rows; | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @return array | ||
|  |      */ | ||
|  |     public function getSpecifix(): array | ||
|  |     { | ||
|  |         return is_array($this->specifix) ? $this->specifix : []; | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @throws FireflyException | ||
|  |      */ | ||
|  |     public function run() | ||
|  |     { | ||
|  |         set_time_limit(0); | ||
|  | 
 | ||
|  |         $this->journals = new Collection; | ||
|  |         $this->map      = $this->data->getMap(); | ||
|  |         $this->roles    = $this->data->getRoles(); | ||
|  |         $this->mapped   = $this->data->getMapped(); | ||
|  |         $this->specifix = $this->data->getSpecifix(); | ||
|  | 
 | ||
|  |         foreach ($this->data->getReader() as $index => $row) { | ||
|  |             if ($this->parseRow($index)) { | ||
|  |                 $this->rows++; | ||
|  |                 $result = $this->importRow($row); | ||
|  |                 if (!($result instanceof TransactionJournal)) { | ||
|  |                     Log::error('Caught error at row #' . $index . ': ' . $result); | ||
|  |                     $this->errors[$index] = $result; | ||
|  |                 } else { | ||
|  |                     $this->imported++; | ||
|  |                     $this->journals->push($result); | ||
|  |                     event(new TransactionJournalStored($result, 0)); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @param Data $data | ||
|  |      */ | ||
|  |     public function setData(Data $data) | ||
|  |     { | ||
|  |         $this->data = $data; | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @return TransactionJournal|string | ||
|  |      */ | ||
|  |     protected function createTransactionJournal() | ||
|  |     { | ||
|  |         $date = $this->importData['date']; | ||
|  |         if (is_null($this->importData['date'])) { | ||
|  |             $date = $this->importData['date-rent']; | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         $transactionType = $this->getTransactionType(); // defaults to deposit
 | ||
|  |         $errors          = new MessageBag; | ||
|  |         $journal         = TransactionJournal::create( | ||
|  |             [ | ||
|  |                 'user_id'                 => Auth::user()->id, | ||
|  |                 'transaction_type_id'     => $transactionType->id, | ||
|  |                 'transaction_currency_id' => $this->importData['currency']->id, | ||
|  |                 'description'             => $this->importData['description'], | ||
|  |                 'completed'               => 0, | ||
|  |                 'date'                    => $date, | ||
|  |                 'bill_id'                 => $this->importData['bill-id'], | ||
|  |             ] | ||
|  |         ); | ||
|  |         if ($journal->getErrors()->count() == 0) { | ||
|  |             // first transaction
 | ||
|  |             $accountId   = $this->importData['asset-account-object']->id; // create first transaction:
 | ||
|  |             $amount      = $this->importData['amount']; | ||
|  |             $transaction = Transaction::create(['transaction_journal_id' => $journal->id, 'account_id' => $accountId, 'amount' => $amount]); | ||
|  |             $errors      = $transaction->getErrors(); | ||
|  | 
 | ||
|  |             // second transaction
 | ||
|  |             $accountId   = $this->importData['opposing-account-object']->id; // create second transaction:
 | ||
|  |             $amount      = bcmul($this->importData['amount'], '-1'); | ||
|  |             $transaction = Transaction::create(['transaction_journal_id' => $journal->id, 'account_id' => $accountId, 'amount' => $amount]); | ||
|  |             $errors      = $transaction->getErrors()->merge($errors); | ||
|  |         } | ||
|  |         if ($errors->count() == 0) { | ||
|  |             $journal->completed = 1; | ||
|  |             $journal->save(); | ||
|  |         } else { | ||
|  |             $text = join(',', $errors->all()); | ||
|  | 
 | ||
|  |             return $text; | ||
|  |         } | ||
|  |         $this->saveBudget($journal); | ||
|  |         $this->saveCategory($journal); | ||
|  |         $this->saveTags($journal); | ||
|  | 
 | ||
|  |         // some debug info:
 | ||
|  |         $journalId = $journal->id; | ||
|  |         $type      = $journal->transaction_type_type ?? $journal->transactionType->type; | ||
|  |         /** @var Account $asset */ | ||
|  |         $asset = $this->importData['asset-account-object']; | ||
|  |         /** @var Account $opposing */ | ||
|  |         $opposing = $this->importData['opposing-account-object']; | ||
|  | 
 | ||
|  |         Log::info('Created journal #' . $journalId . ' of type ' . $type . '!'); | ||
|  |         Log::info('Asset account #' . $asset->id . ' lost/gained: ' . $this->importData['amount']); | ||
|  |         Log::info($opposing->accountType->type . ' #' . $opposing->id . ' lost/gained: ' . bcmul($this->importData['amount'], '-1')); | ||
|  | 
 | ||
|  |         return $journal; | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @return TransactionType | ||
|  |      */ | ||
|  |     protected function getTransactionType() | ||
|  |     { | ||
|  |         $transactionType = TransactionType::where('type', TransactionType::DEPOSIT)->first(); | ||
|  |         if ($this->importData['amount'] < 0) { | ||
|  |             $transactionType = TransactionType::where('type', TransactionType::WITHDRAWAL)->first(); | ||
|  |         } | ||
|  | 
 | ||
|  |         if (in_array($this->importData['opposing-account-object']->accountType->type, ['Asset account', 'Default account'])) { | ||
|  |             $transactionType = TransactionType::where('type', TransactionType::TRANSFER)->first(); | ||
|  |         } | ||
|  | 
 | ||
|  |         return $transactionType; | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @param array $row | ||
|  |      * | ||
|  |      * @throws FireflyException | ||
|  |      * @return string|bool | ||
|  |      */ | ||
|  |     protected function importRow(array $row) | ||
|  |     { | ||
|  | 
 | ||
|  |         $data = $this->getFiller(); // These fields are necessary to create a new transaction journal. Some are optional
 | ||
|  |         foreach ($row as $index => $value) { | ||
|  |             $role  = $this->roles[$index] ?? '_ignore'; | ||
|  |             $class = config('csv.roles.' . $role . '.converter'); | ||
|  |             $field = config('csv.roles.' . $role . '.field'); | ||
|  | 
 | ||
|  | 
 | ||
|  |             // here would be the place where preprocessors would fire.
 | ||
|  | 
 | ||
|  |             /** @var ConverterInterface $converter */ | ||
|  |             $converter = app('FireflyIII\Helpers\Csv\Converter\\' . $class); | ||
|  |             $converter->setData($data); // the complete array so far.
 | ||
|  |             $converter->setField($field); | ||
|  |             $converter->setIndex($index); | ||
|  |             $converter->setMapped($this->mapped); | ||
|  |             $converter->setValue($value); | ||
|  |             $data[$field] = $converter->convert(); | ||
|  |         } | ||
|  |         // move to class vars.
 | ||
|  |         $this->importData = $data; | ||
|  |         $this->importRow  = $row; | ||
|  |         unset($data, $row); | ||
|  |         // post processing and validating.
 | ||
|  |         $this->postProcess(); | ||
|  |         $result = $this->validateData(); | ||
|  | 
 | ||
|  |         if (!($result === true)) { | ||
|  |             return $result; // return error.
 | ||
|  |         } | ||
|  |         $journal = $this->createTransactionJournal(); | ||
|  | 
 | ||
|  |         return $journal; | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @param int $index | ||
|  |      * | ||
|  |      * @return bool | ||
|  |      */ | ||
|  |     protected function parseRow(int $index) | ||
|  |     { | ||
|  |         return (($this->data->hasHeaders() && $index >= 1) || !$this->data->hasHeaders()); | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Row denotes the original data. | ||
|  |      * | ||
|  |      * @return void | ||
|  |      */ | ||
|  |     protected function postProcess() | ||
|  |     { | ||
|  |         // do bank specific fixes (must be enabled but now all of them.
 | ||
|  | 
 | ||
|  |         foreach ($this->getSpecifix() as $className) { | ||
|  |             /** @var SpecifixInterface $specifix */ | ||
|  |             $specifix = app('FireflyIII\Helpers\Csv\Specifix\\' . $className); | ||
|  |             if ($specifix->getProcessorType() == SpecifixInterface::POST_PROCESSOR) { | ||
|  |                 $specifix->setData($this->importData); | ||
|  |                 $specifix->setRow($this->importRow); | ||
|  |                 $this->importData = $specifix->fix(); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         $set = config('csv.post_processors'); | ||
|  |         foreach ($set as $className) { | ||
|  |             /** @var PostProcessorInterface $postProcessor */ | ||
|  |             $postProcessor = app('FireflyIII\Helpers\Csv\PostProcessing\\' . $className); | ||
|  |             $array         = $this->importData ?? []; | ||
|  |             $postProcessor->setData($array); | ||
|  |             $this->importData = $postProcessor->process(); | ||
|  |         } | ||
|  | 
 | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @param TransactionJournal $journal | ||
|  |      */ | ||
|  |     protected function saveBudget(TransactionJournal $journal) | ||
|  |     { | ||
|  |         // add budget:
 | ||
|  |         if (!is_null($this->importData['budget'])) { | ||
|  |             $journal->budgets()->save($this->importData['budget']); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @param TransactionJournal $journal | ||
|  |      */ | ||
|  |     protected function saveCategory(TransactionJournal $journal) | ||
|  |     { | ||
|  |         // add category:
 | ||
|  |         if (!is_null($this->importData['category'])) { | ||
|  |             $journal->categories()->save($this->importData['category']); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @param TransactionJournal $journal | ||
|  |      */ | ||
|  |     protected function saveTags(TransactionJournal $journal) | ||
|  |     { | ||
|  |         if (!is_null($this->importData['tags'])) { | ||
|  |             foreach ($this->importData['tags'] as $tag) { | ||
|  |                 $journal->tags()->save($tag); | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * | ||
|  |      * @return bool|string | ||
|  |      */ | ||
|  |     protected function validateData() | ||
|  |     { | ||
|  |         $date     = $this->importData['date'] ?? null; | ||
|  |         $rentDate = $this->importData['date-rent'] ?? null; | ||
|  |         if (is_null($date) && is_null($rentDate)) { | ||
|  |             return 'No date value for this row.'; | ||
|  |         } | ||
|  |         if (is_null($this->importData['opposing-account-object'])) { | ||
|  |             return 'Opposing account is null'; | ||
|  |         } | ||
|  | 
 | ||
|  |         if (!($this->importData['asset-account-object'] instanceof Account)) { | ||
|  |             return 'No asset account to import into.'; | ||
|  |         } | ||
|  | 
 | ||
|  |         return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * @return array | ||
|  |      */ | ||
|  |     private function getFiller() | ||
|  |     { | ||
|  |         $filler = []; | ||
|  |         foreach (config('csv.roles') as $role) { | ||
|  |             if (isset($role['field'])) { | ||
|  |                 $fieldName          = $role['field']; | ||
|  |                 $filler[$fieldName] = null; | ||
|  |             } | ||
|  |         } | ||
|  |         // some extra's:
 | ||
|  |         $filler['bill-id']                 = null; | ||
|  |         $filler['opposing-account-object'] = null; | ||
|  |         $filler['asset-account-object']    = null; | ||
|  |         $filler['amount-modifier']         = '1'; | ||
|  | 
 | ||
|  |         return $filler; | ||
|  | 
 | ||
|  |     } | ||
|  | 
 | ||
|  | } |