You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							640 lines
						
					
					
						
							27 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							640 lines
						
					
					
						
							27 KiB
						
					
					
				| <?php | |
| /* | |
|  * AutoImports.php | |
|  * Copyright (c) 2021 james@firefly-iii.org | |
|  * | |
|  * This file is part of the Firefly III Data Importer | |
|  * (https://github.com/firefly-iii/data-importer). | |
|  * | |
|  * This program is free software: you can redistribute it and/or modify | |
|  * it under the terms of the GNU Affero General Public License as | |
|  * published by the Free Software Foundation, either version 3 of the | |
|  * License, or (at your option) any later version. | |
|  * | |
|  * This program 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 Affero General Public License for more details. | |
|  * | |
|  * You should have received a copy of the GNU Affero General Public License | |
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | |
|  */ | |
| 
 | |
| declare(strict_types=1); | |
| 
 | |
| namespace App\Console; | |
| 
 | |
| use App\Events\ImportedTransactions; | |
| use App\Exceptions\ImporterErrorException; | |
| use App\Services\Camt\Conversion\RoutineManager as CamtRoutineManager; | |
| use App\Services\CSV\Conversion\RoutineManager as CSVRoutineManager; | |
| use App\Services\Nordigen\Conversion\RoutineManager as NordigenRoutineManager; | |
| use App\Services\Nordigen\Model\Account; | |
| use App\Services\Nordigen\Model\Balance; | |
| use App\Services\Shared\Authentication\SecretManager; | |
| use App\Services\Shared\Configuration\Configuration; | |
| use App\Services\Shared\Conversion\ConversionStatus; | |
| use App\Services\Shared\Conversion\RoutineStatusManager; | |
| use App\Services\Shared\File\FileContentSherlock; | |
| use App\Services\Shared\Import\Routine\RoutineManager; | |
| use App\Services\Shared\Import\Status\SubmissionStatus; | |
| use App\Services\Shared\Import\Status\SubmissionStatusManager; | |
| use App\Services\Spectre\Conversion\RoutineManager as SpectreRoutineManager; | |
| use Carbon\Carbon; | |
| use GrumpyDictator\FFIIIApiSupport\Exceptions\ApiHttpException; | |
| use GrumpyDictator\FFIIIApiSupport\Model\Account as LocalAccount; | |
| use GrumpyDictator\FFIIIApiSupport\Request\GetAccountRequest; | |
| use Illuminate\Contracts\Filesystem\FileNotFoundException; | |
| use Illuminate\Support\Facades\Log; | |
| 
 | |
| /** | |
|  * Trait AutoImports | |
|  */ | |
| trait AutoImports | |
| { | |
|     protected array  $conversionErrors   = []; | |
|     protected array  $conversionMessages = []; | |
|     protected array  $conversionWarnings = []; | |
|     protected string $identifier; | |
|     protected array  $importErrors       = []; | |
|     protected array  $importMessages     = []; | |
|     protected array  $importWarnings     = []; | |
|     protected array  $importerAccounts   = []; | |
| 
 | |
|     private function getFiles(string $directory): array | |
|     { | |
|         $ignore = ['.', '..']; | |
| 
 | |
|         if ('' === $directory) { | |
|             $this->error(sprintf('Directory "%s" is empty or invalid.', $directory)); | |
| 
 | |
|             return []; | |
|         } | |
|         $array  = scandir($directory); | |
|         if (!is_array($array)) { | |
|             $this->error(sprintf('Directory "%s" is empty or invalid.', $directory)); | |
| 
 | |
|             return []; | |
|         } | |
|         $files  = array_diff($array, $ignore); | |
|         $return = []; | |
|         foreach ($files as $file) { | |
|             // import importable file with JSON companion | |
|             // TODO may also need to be able to detect other file types. Or: detect any file with accompanying json file | |
|             if ('csv' === $this->getExtension($file) && $this->hasJsonConfiguration($directory, $file)) { | |
|                 $return[] = $file; | |
|             } | |
| 
 | |
|             if ('xml' === $this->getExtension($file) && $this->hasJsonConfiguration($directory, $file)) { | |
|                 $return[] = $file; | |
|             } | |
| 
 | |
|             // import JSON with no importable file. | |
|             // TODO must detect json files without accompanying camt/csv/whatever file. | |
|             if ('json' === $this->getExtension($file) && !$this->hasCsvFile($directory, $file)) { | |
|                 $return[] = $file; | |
|             } | |
|         } | |
| 
 | |
|         return $return; | |
|     } | |
| 
 | |
|     private function getExtension(string $file): string | |
|     { | |
|         $parts = explode('.', $file); | |
|         if (1 === count($parts)) { | |
|             return ''; | |
|         } | |
| 
 | |
|         return strtolower($parts[count($parts) - 1]); | |
|     } | |
| 
 | |
|     /** | |
|      * This method only works on files with an extension with exactly three letters | |
|      * (ie. "csv", "xml"). | |
|      */ | |
|     private function hasJsonConfiguration(string $directory, string $file): bool | |
|     { | |
|         $short    = substr($file, 0, -4); | |
|         $jsonFile = sprintf('%s.json', $short); | |
|         $fullJson = sprintf('%s/%s', $directory, $jsonFile); | |
| 
 | |
|         if (!file_exists($fullJson)) { | |
|             $hasFallbackConfig = $this->hasFallbackConfig($directory); | |
|             if ($hasFallbackConfig) { | |
|                 $this->line('Found fallback configuration file, which will be used for this file.'); | |
| 
 | |
|                 return true; | |
|             } | |
|             $this->warn(sprintf('Cannot find JSON file "%s" nor fallback file expected to go with file "%s". This file will be ignored.', $jsonFile, $file)); | |
| 
 | |
|             return false; | |
|         } | |
| 
 | |
|         return true; | |
|     } | |
| 
 | |
|     private function hasFallbackConfig(string $directory): bool | |
|     { | |
|         if (false === config('importer.fallback_in_dir')) { | |
|             return false; | |
|         } | |
|         $configJsonFile = sprintf('%s/%s', $directory, config('importer.fallback_configuration')); | |
|         if (file_exists($configJsonFile) && is_readable($configJsonFile)) { | |
|             $content = file_get_contents($configJsonFile); | |
| 
 | |
|             return json_validate($content); | |
|         } | |
| 
 | |
|         return false; | |
|     } | |
| 
 | |
|     /** | |
|      * TODO this function must be more universal. | |
|      */ | |
|     private function hasCsvFile(string $directory, string $file): bool | |
|     { | |
|         $short    = substr($file, 0, -5); | |
|         $csvFile  = sprintf('%s.csv', $short); | |
|         $fullJson = sprintf('%s/%s', $directory, $csvFile); | |
|         if (!file_exists($fullJson)) { | |
|             return false; | |
|         } | |
| 
 | |
|         return true; | |
|     } | |
| 
 | |
|     private function importFiles(string $directory, array $files): array | |
|     { | |
|         $exitCodes = []; | |
| 
 | |
|         /** @var string $file */ | |
|         foreach ($files as $file) { | |
|             $key                      = sprintf('%s/%s', $directory, $file); | |
| 
 | |
|             try { | |
|                 $exitCodes[$key] = $this->importFile($directory, $file); | |
|             } catch (ImporterErrorException $e) { | |
|                 app('log')->error(sprintf('Could not complete import from file "%s".', $file)); | |
|                 app('log')->error($e->getMessage()); | |
|                 $exitCodes[$key] = 1; | |
|             } | |
|             // report has already been sent. Reset errors and continue. | |
|             $this->conversionErrors   = []; | |
|             $this->conversionMessages = []; | |
|             $this->conversionWarnings = []; | |
|             $this->importErrors       = []; | |
|             $this->importMessages     = []; | |
|             $this->importWarnings     = []; | |
|         } | |
| 
 | |
|         return $exitCodes; | |
|     } | |
| 
 | |
|     /** | |
|      * @throws ImporterErrorException | |
|      */ | |
|     private function importFile(string $directory, string $file): int | |
|     { | |
|         app('log')->debug(sprintf('ImportFile: directory "%s"', $directory)); | |
|         app('log')->debug(sprintf('ImportFile: file      "%s"', $file)); | |
|         $importableFile    = sprintf('%s/%s', $directory, $file); | |
|         $jsonFile          = sprintf('%s/%s.json', $directory, substr($file, 0, -5)); | |
|         $fallbackJsonFile  = sprintf('%s/%s', $directory, config('importer.fallback_configuration')); | |
| 
 | |
|         // TODO not yet sure why the distinction is necessary. | |
|         // TODO this may also be necessary for camt files. | |
|         if ('csv' === $this->getExtension($file)) { | |
|             $jsonFile = sprintf('%s/%s.json', $directory, substr($file, 0, -4)); | |
|         } | |
|         // same for XML files. | |
|         if ('xml' === $this->getExtension($file)) { | |
|             $jsonFile = sprintf('%s/%s.json', $directory, substr($file, 0, -4)); | |
|         } | |
|         $jsonFileExists    = file_exists($jsonFile); | |
|         $hasFallbackConfig = $this->hasFallbackConfig($directory); | |
| 
 | |
|         // Should not happen | |
|         if (!$jsonFileExists && !$hasFallbackConfig) { | |
|             $this->error(sprintf('No JSON configuration found. Checked for both "%s" and "%s"', $jsonFile, $fallbackJsonFile)); | |
| 
 | |
|             return 68; | |
|         } | |
| 
 | |
|         $jsonFile          = $jsonFileExists ? $jsonFile : $fallbackJsonFile; | |
| 
 | |
|         app('log')->debug(sprintf('ImportFile: importable "%s"', $importableFile)); | |
|         app('log')->debug(sprintf('ImportFile: JSON       "%s"', $jsonFile)); | |
| 
 | |
|         // do JSON check | |
|         $jsonResult        = $this->verifyJSON($jsonFile); | |
|         if (false === $jsonResult) { | |
|             $message = sprintf('The importer can\'t import %s: could not decode the JSON in config file %s.', $importableFile, $jsonFile); | |
|             $this->error($message); | |
| 
 | |
|             return 69; | |
|         } | |
|         $configuration     = Configuration::fromArray(json_decode(file_get_contents($jsonFile), true)); | |
| 
 | |
|         // sanity check. If the importableFile is a .json file, and it parses as valid json, don't import it: | |
|         if ('file' === $configuration->getFlow() && str_ends_with(strtolower($importableFile), '.json') && $this->verifyJSON($importableFile)) { | |
|             app('log')->warning('Almost tried to import a JSON file as a file lol. Skip it.'); | |
| 
 | |
|             // don't report this. | |
|             return 0; | |
|         } | |
| 
 | |
|         $configuration->updateDateRange(); | |
|         $this->line(sprintf('Going to convert from file %s using configuration %s and flow "%s".', $importableFile, $jsonFile, $configuration->getFlow())); | |
| 
 | |
|         // this is it! | |
|         $this->startConversion($configuration, $importableFile); | |
|         $this->reportConversion(); | |
| 
 | |
|         // crash here if the conversion failed. | |
|         if (0 !== count($this->conversionErrors)) { | |
|             $this->error(sprintf('Too many errors in the data conversion (%d), exit.', count($this->conversionErrors))); | |
| 
 | |
|             // report about it anyway: | |
|             event( | |
|                 new ImportedTransactions( | |
|                     array_merge($this->conversionMessages, $this->importMessages), | |
|                     array_merge($this->conversionWarnings, $this->importWarnings), | |
|                     array_merge($this->conversionErrors, $this->importErrors) | |
|                 ) | |
|             ); | |
| 
 | |
|             return 72; | |
|         } | |
| 
 | |
|         $this->line(sprintf('Done converting from file %s using configuration %s.', $importableFile, $jsonFile)); | |
|         $this->startImport($configuration); | |
|         $this->reportImport(); | |
|         $this->reportBalanceDifferences($configuration); | |
| 
 | |
|         $this->line('Done!'); | |
| 
 | |
|         // merge things: | |
|         $messages          = array_merge($this->importMessages, $this->conversionMessages); | |
|         $warnings          = array_merge($this->importWarnings, $this->conversionWarnings); | |
|         $errors            = array_merge($this->importErrors, $this->conversionErrors); | |
|         event(new ImportedTransactions($messages, $warnings, $errors)); | |
| 
 | |
|         if (count($this->importErrors) > 0) { | |
|             return 1; | |
|         } | |
|         if (0 === count($messages) && 0 === count($warnings) && 0 === count($errors)) { | |
|             return 73; | |
|         } | |
| 
 | |
|         return 0; | |
|     } | |
| 
 | |
|     /** | |
|      * @throws ImporterErrorException | |
|      */ | |
|     private function startConversion(Configuration $configuration, string $importableFile): void | |
|     { | |
|         $this->conversionMessages = []; | |
|         $this->conversionWarnings = []; | |
|         $this->conversionErrors   = []; | |
|         $flow                     = $configuration->getFlow(); | |
| 
 | |
|         app('log')->debug(sprintf('Now in %s', __METHOD__)); | |
| 
 | |
|         if ('' === $importableFile && 'file' === $flow) { | |
|             $this->warn('Importable file path is empty. That means there is no importable file to import.'); | |
| 
 | |
|             exit(1); | |
|         } | |
| 
 | |
|         $manager                  = null; | |
|         if ('file' === $flow) { | |
|             $contentType = $configuration->getContentType(); | |
|             if ('unknown' === $contentType) { | |
|                 app('log')->debug('Content type is "unknown" in startConversion(), detect it.'); | |
|                 $detector    = new FileContentSherlock(); | |
|                 $contentType = $detector->detectContentType($importableFile); | |
|             } | |
|             if ('unknown' === $contentType || 'csv' === $contentType) { | |
|                 app('log')->debug(sprintf('Content type is "%s" in startConversion(), use the CSV routine.', $contentType)); | |
|                 $manager          = new CSVRoutineManager(null); | |
|                 $this->identifier = $manager->getIdentifier(); | |
|                 $manager->setContent(file_get_contents($importableFile)); | |
|             } | |
|             if ('camt' === $contentType) { | |
|                 app('log')->debug('Content type is "camt" in startConversion(), use the CAMT routine.'); | |
|                 $manager          = new CamtRoutineManager(null); | |
|                 $this->identifier = $manager->getIdentifier(); | |
|                 $manager->setContent(file_get_contents($importableFile)); | |
|             } | |
|         } | |
|         if ('nordigen' === $flow) { | |
|             $manager          = new NordigenRoutineManager(null); | |
|             $this->identifier = $manager->getIdentifier(); | |
|         } | |
|         if ('spectre' === $flow) { | |
|             $manager          = new SpectreRoutineManager(null); | |
|             $this->identifier = $manager->getIdentifier(); | |
|         } | |
|         if (null === $manager) { | |
|             $this->error(sprintf('There is no support for flow "%s"', $flow)); | |
| 
 | |
|             exit(1); | |
|         } | |
| 
 | |
|         RoutineStatusManager::startOrFindConversion($this->identifier); | |
|         RoutineStatusManager::setConversionStatus(ConversionStatus::CONVERSION_RUNNING, $this->identifier); | |
| 
 | |
|         // then push stuff into the routine: | |
|         $manager->setConfiguration($configuration); | |
|         $transactions             = []; | |
| 
 | |
|         try { | |
|             $transactions = $manager->start(); | |
|         } catch (ImporterErrorException $e) { | |
|             app('log')->error($e->getMessage()); | |
|             RoutineStatusManager::setConversionStatus(ConversionStatus::CONVERSION_ERRORED, $this->identifier); | |
|             $this->conversionMessages = $manager->getAllMessages(); | |
|             $this->conversionWarnings = $manager->getAllWarnings(); | |
|             $this->conversionErrors   = $manager->getAllErrors(); | |
|         } | |
|         if (0 === count($transactions)) { | |
|             app('log')->error('[a] Zero transactions!'); | |
|             RoutineStatusManager::setConversionStatus(ConversionStatus::CONVERSION_DONE, $this->identifier); | |
|             $this->conversionMessages = $manager->getAllMessages(); | |
|             $this->conversionWarnings = $manager->getAllWarnings(); | |
|             $this->conversionErrors   = $manager->getAllErrors(); | |
|         } | |
| 
 | |
|         // save transactions in 'jobs' directory under the same key as the conversion thing. | |
|         $disk                     = \Storage::disk('jobs'); | |
| 
 | |
|         try { | |
|             $disk->put(sprintf('%s.json', $this->identifier), json_encode($transactions, JSON_PRETTY_PRINT | JSON_THROW_ON_ERROR)); | |
|         } catch (\JsonException $e) { | |
|             app('log')->error(sprintf('JSON exception: %s', $e->getMessage())); | |
|             RoutineStatusManager::setConversionStatus(ConversionStatus::CONVERSION_ERRORED, $this->identifier); | |
|             $this->conversionMessages = $manager->getAllMessages(); | |
|             $this->conversionWarnings = $manager->getAllWarnings(); | |
|             $this->conversionErrors   = $manager->getAllErrors(); | |
|             $transactions             = []; | |
|         } | |
| 
 | |
|         if (count($transactions) > 0) { | |
|             // set done: | |
|             RoutineStatusManager::setConversionStatus(ConversionStatus::CONVERSION_DONE, $this->identifier); | |
| 
 | |
|             $this->conversionMessages = $manager->getAllMessages(); | |
|             $this->conversionWarnings = $manager->getAllWarnings(); | |
|             $this->conversionErrors   = $manager->getAllErrors(); | |
|         } | |
|         $this->importerAccounts   = $manager->getServiceAccounts(); | |
|     } | |
| 
 | |
|     private function reportConversion(): void | |
|     { | |
|         $list = [ | |
|             'info'  => $this->conversionMessages, | |
|             'warn'  => $this->conversionWarnings, | |
|             'error' => $this->conversionErrors, | |
|         ]; | |
|         foreach ($list as $func => $set) { | |
|             /** | |
|              * @var int   $index | |
|              * @var array $messages | |
|              */ | |
|             foreach ($set as $index => $messages) { | |
|                 if (count($messages) > 0) { | |
|                     foreach ($messages as $message) { | |
|                         $this->{$func}(sprintf('Conversion index (%s) %d: %s', $func, $index, $message)); // @phpstan-ignore-line | |
|                     } | |
|                 } | |
|             } | |
|         } | |
|     } | |
| 
 | |
|     private function startImport(Configuration $configuration): void | |
|     { | |
|         app('log')->debug(sprintf('Now at %s', __METHOD__)); | |
|         $routine              = new RoutineManager($this->identifier); | |
|         SubmissionStatusManager::startOrFindSubmission($this->identifier); | |
|         $disk                 = \Storage::disk('jobs'); | |
|         $fileName             = sprintf('%s.json', $this->identifier); | |
| 
 | |
|         // get files from disk: | |
|         if (!$disk->has($fileName)) { | |
|             SubmissionStatusManager::setSubmissionStatus(SubmissionStatus::SUBMISSION_ERRORED, $this->identifier); | |
|             $message              = sprintf('File "%s" not found, cannot continue.', $fileName); | |
|             $this->error($message); | |
|             SubmissionStatusManager::addError($this->identifier, 0, $message); | |
|             $this->importMessages = $routine->getAllMessages(); | |
|             $this->importWarnings = $routine->getAllWarnings(); | |
|             $this->importErrors   = $routine->getAllErrors(); | |
| 
 | |
|             return; | |
|         } | |
| 
 | |
|         try { | |
|             $json         = $disk->get($fileName); | |
|             $transactions = json_decode($json, true, 512, JSON_THROW_ON_ERROR); | |
|             app('log')->debug(sprintf('Found %d transactions on the drive.', count($transactions))); | |
|         } catch (FileNotFoundException|\JsonException $e) { | |
|             SubmissionStatusManager::setSubmissionStatus(SubmissionStatus::SUBMISSION_ERRORED, $this->identifier); | |
|             $message              = sprintf('File "%s" could not be decoded, cannot continue..', $fileName); | |
|             $this->error($message); | |
|             SubmissionStatusManager::addError($this->identifier, 0, $message); | |
|             $this->importMessages = $routine->getAllMessages(); | |
|             $this->importWarnings = $routine->getAllWarnings(); | |
|             $this->importErrors   = $routine->getAllErrors(); | |
| 
 | |
|             return; | |
|         } | |
|         if (0 === count($transactions)) { | |
|             SubmissionStatusManager::setSubmissionStatus(SubmissionStatus::SUBMISSION_DONE, $this->identifier); | |
|             app('log')->error('No transactions in array, there is nothing to import.'); | |
|             $this->importMessages = $routine->getAllMessages(); | |
|             $this->importWarnings = $routine->getAllWarnings(); | |
|             $this->importErrors   = $routine->getAllErrors(); | |
| 
 | |
|             return; | |
|         } | |
| 
 | |
|         $routine->setTransactions($transactions); | |
| 
 | |
|         SubmissionStatusManager::setSubmissionStatus(SubmissionStatus::SUBMISSION_RUNNING, $this->identifier); | |
| 
 | |
|         // then push stuff into the routine: | |
|         $routine->setConfiguration($configuration); | |
| 
 | |
|         try { | |
|             $routine->start(); | |
|         } catch (ImporterErrorException $e) { | |
|             app('log')->error($e->getMessage()); | |
|             SubmissionStatusManager::setSubmissionStatus(SubmissionStatus::SUBMISSION_ERRORED, $this->identifier); | |
|             SubmissionStatusManager::addError($this->identifier, 0, $e->getMessage()); | |
|             $this->importMessages = $routine->getAllMessages(); | |
|             $this->importWarnings = $routine->getAllWarnings(); | |
|             $this->importErrors   = $routine->getAllErrors(); | |
| 
 | |
|             return; | |
|         } | |
| 
 | |
|         // set done: | |
|         SubmissionStatusManager::setSubmissionStatus(SubmissionStatus::SUBMISSION_DONE, $this->identifier); | |
|         $this->importMessages = $routine->getAllMessages(); | |
|         $this->importWarnings = $routine->getAllWarnings(); | |
|         $this->importErrors   = $routine->getAllErrors(); | |
|     } | |
| 
 | |
|     private function reportImport(): void | |
|     { | |
|         $list = [ | |
|             'info'  => $this->importMessages, | |
|             'warn'  => $this->importWarnings, | |
|             'error' => $this->importErrors, | |
|         ]; | |
| 
 | |
|         $this->info(sprintf('There are %d message(s)', count($this->importMessages))); | |
|         $this->info(sprintf('There are %d warning(s)', count($this->importWarnings))); | |
|         $this->info(sprintf('There are %d error(s)', count($this->importErrors))); | |
| 
 | |
|         foreach ($list as $func => $set) { | |
|             /** | |
|              * @var int   $index | |
|              * @var array $messages | |
|              */ | |
|             foreach ($set as $index => $messages) { | |
|                 if (count($messages) > 0) { | |
|                     foreach ($messages as $message) { | |
|                         $this->{$func}(sprintf('Import index %d: %s', $index, $message)); // @phpstan-ignore-line | |
|                     } | |
|                 } | |
|             } | |
|         } | |
|     } | |
| 
 | |
|     private function reportBalanceDifferences(Configuration $configuration): void | |
|     { | |
|         if ('nordigen' !== $configuration->getFlow()) { | |
|             return; | |
|         } | |
|         $count         = count($this->importerAccounts); | |
|         $localAccounts = $configuration->getAccounts(); | |
|         $url           = SecretManager::getBaseUrl(); | |
|         $token         = SecretManager::getAccessToken(); | |
|         Log::debug(sprintf('The importer has collected %d account(s) to report the balance difference on.', $count)); | |
| 
 | |
|         /** @var Account $account */ | |
|         foreach ($this->importerAccounts as $account) { | |
|             // check if account exists: | |
|             if (!array_key_exists($account->getIdentifier(), $localAccounts)) { | |
|                 Log::debug(sprintf('GoCardless account "%s" (IBAN "%s") is not being imported, so skipped.', $account->getIdentifier(), $account->getIban())); | |
| 
 | |
|                 continue; | |
|             } | |
|             // local account ID exists, we can check the balance over at Firefly III. | |
|             $accountId      = $localAccounts[$account->getIdentifier()]; | |
|             $accountRequest = new GetAccountRequest($url, $token); | |
|             $accountRequest->setVerify(config('importer.connection.verify')); | |
|             $accountRequest->setTimeOut(config('importer.connection.timeout')); | |
|             $accountRequest->setId($accountId); | |
| 
 | |
|             try { | |
|                 $result = $accountRequest->get(); | |
|             } catch (ApiHttpException $e) { | |
|                 app('log')->error('Could not get Firefly III account for balance check. Will ignore this issue.'); | |
|                 app('log')->debug($e->getMessage()); | |
| 
 | |
|                 continue; | |
|             } | |
| 
 | |
|             /** @var LocalAccount $localAccount */ | |
|             $localAccount   = $result->getAccount(); | |
| 
 | |
|             $this->reportBalanceDifference($account, $localAccount); | |
|         } | |
|     } | |
| 
 | |
|     private function reportBalanceDifference(Account $account, LocalAccount $localAccount): void | |
|     { | |
|         Log::debug(sprintf('Report balance difference between GoCardless account "%s" and Firefly III account #%d.', $account->getIdentifier(), $localAccount->id)); | |
|         app('log')->debug(sprintf('GoCardless account has %d balance entry (entries)', count($account->getBalances()))); | |
| 
 | |
|         /** @var Balance $balance */ | |
|         foreach ($account->getBalances() as $index => $balance) { | |
|             app('log')->debug(sprintf('Now comparing balance entry "%s" (#%d of %d)', $balance->type, $index + 1, count($account->getBalances()))); | |
|             $this->reportSingleDifference($account, $localAccount, $balance); | |
|         } | |
|     } | |
| 
 | |
|     private function reportSingleDifference(Account $account, LocalAccount $localAccount, Balance $balance): void | |
|     { | |
|         // compare currencies, and warn if necessary. | |
|         if ($balance->currency !== $localAccount->currencyCode) { | |
|             app('log')->warning(sprintf('GoCardless account "%s" has currency %s, Firefly III account #%d uses %s.', $account->getIdentifier(), $localAccount->id, $balance->currency, $localAccount->currencyCode)); | |
|             $this->line(sprintf('Balance comparison (%s): Firefly III account #%d: Currency mismatch', $balance->type, $localAccount->id)); | |
|         } | |
| 
 | |
|         // compare dates, warn | |
|         $date      = Carbon::parse($balance->date); | |
|         $localDate = Carbon::parse($localAccount->currentBalanceDate); | |
|         if (!$date->isSameDay($localDate)) { | |
|             app('log')->warning(sprintf('GoCardless balance is from day %s, Firefly III account from %s.', $date->format('Y-m-d'), $date->format('Y-m-d'))); | |
|             $this->line(sprintf('Balance comparison (%s): Firefly III account #%d: Date mismatch', $balance->type, $localAccount->id)); | |
|         } | |
| 
 | |
|         // compare balance, warn (also a message) | |
|         app('log')->debug(sprintf('Comparing %s and %s', $balance->amount, $localAccount->currentBalance)); | |
|         if (0 !== bccomp($balance->amount, $localAccount->currentBalance)) { | |
|             app('log')->warning(sprintf('GoCardless balance is %s, Firefly III balance is %s.', $balance->amount, $localAccount->currentBalance)); | |
|             $this->line(sprintf('Balance comparison (%s): Firefly III account #%d: GoCardless reports %s %s, Firefly III reports %s %d', $balance->type, $localAccount->id, $balance->currency, $balance->amount, $localAccount->currencyCode, $localAccount->currentBalance)); | |
|         } | |
|         if (0 === bccomp($balance->amount, $localAccount->currentBalance)) { | |
|             $this->line(sprintf('Balance comparison (%s): Firefly III account #%d: Balance OK', $balance->type, $localAccount->id)); | |
|         } | |
|     } | |
| 
 | |
|     /** | |
|      * @throws ImporterErrorException | |
|      */ | |
|     private function importUpload(string $jsonFile, string $importableFile): void | |
|     { | |
|         // do JSON check | |
|         $jsonResult    = $this->verifyJSON($jsonFile); | |
|         if (false === $jsonResult) { | |
|             $message = sprintf('The importer can\'t import %s: could not decode the JSON in config file %s.', $importableFile, $jsonFile); | |
|             $this->error($message); | |
| 
 | |
|             return; | |
|         } | |
|         $configuration = Configuration::fromArray(json_decode(file_get_contents($jsonFile), true)); | |
|         $configuration->updateDateRange(); | |
| 
 | |
|         $this->line(sprintf('Going to convert from file "%s" using configuration "%s" and flow "%s".', $importableFile, $jsonFile, $configuration->getFlow())); | |
| 
 | |
|         // this is it! | |
|         $this->startConversion($configuration, $importableFile); | |
|         $this->reportConversion(); | |
| 
 | |
|         // crash here if the conversion failed. | |
|         if (0 !== count($this->conversionErrors)) { | |
|             $this->error(sprintf('Too many errors in the data conversion (%d), exit.', count($this->conversionErrors))); | |
| 
 | |
|             throw new ImporterErrorException('Too many errors in the data conversion.'); | |
|         } | |
| 
 | |
|         $this->line(sprintf('Done converting from file %s using configuration %s.', $importableFile, $jsonFile)); | |
|         $this->startImport($configuration); | |
|         $this->reportImport(); | |
| 
 | |
|         $this->line('Done!'); | |
|         event( | |
|             new ImportedTransactions( | |
|                 array_merge($this->conversionMessages, $this->importMessages), | |
|                 array_merge($this->conversionWarnings, $this->importWarnings), | |
|                 array_merge($this->conversionErrors, $this->importErrors) | |
|             ) | |
|         ); | |
|     } | |
| }
 |