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.
 
 
 
 
 

164 lines
6.1 KiB

<?php
/*
* PseudoTransactionProcessor.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\Services\CSV\Conversion\Routine;
use App\Exceptions\ImporterErrorException;
use App\Services\CSV\Conversion\Task\AbstractTask;
use App\Services\Shared\Authentication\SecretManager;
use App\Services\Shared\Conversion\ProgressInformation;
use App\Support\RequestCache;
use GrumpyDictator\FFIIIApiSupport\Exceptions\ApiHttpException;
use GrumpyDictator\FFIIIApiSupport\Model\Account;
use GrumpyDictator\FFIIIApiSupport\Model\TransactionCurrency;
use GrumpyDictator\FFIIIApiSupport\Request\GetAccountRequest;
use GrumpyDictator\FFIIIApiSupport\Request\GetCurrencyRequest;
use GrumpyDictator\FFIIIApiSupport\Response\GetAccountResponse;
use GrumpyDictator\FFIIIApiSupport\Response\GetCurrencyResponse;
use Illuminate\Support\Facades\Log;
/**
* Class PseudoTransactionProcessor
*/
class PseudoTransactionProcessor
{
use ProgressInformation;
private Account $defaultAccount;
private TransactionCurrency $defaultCurrency;
private array $tasks;
/**
* PseudoTransactionProcessor constructor.
*
* @throws ImporterErrorException
*/
public function __construct(?int $defaultAccountId)
{
$this->tasks = config('csv.transaction_tasks');
$this->getDefaultAccount($defaultAccountId);
$this->getDefaultCurrency();
}
/**
* @throws ImporterErrorException
*/
private function getDefaultAccount(?int $accountId): void
{
$url = SecretManager::getBaseUrl();
$token = SecretManager::getAccessToken();
$cacheKey = sprintf('%s-%s-%s', $url, $accountId, 'getDefaultAccount');
if (null !== $accountId) {
// in cache perhaps?
$inCache = RequestCache::has($cacheKey, $token);
if ($inCache) {
$this->defaultAccount = RequestCache::get($cacheKey, $token);
return;
}
$accountRequest = new GetAccountRequest($url, $token);
$accountRequest->setVerify(config('importer.connection.verify'));
$accountRequest->setTimeOut(config('importer.connection.timeout'));
$accountRequest->setId($accountId);
try {
/** @var GetAccountResponse $result */
$result = $accountRequest->get();
} catch (ApiHttpException $e) {
Log::error(sprintf('[%s]: %s', config('importer.version'), $e->getMessage()));
throw new ImporterErrorException(sprintf('The default account in your configuration file (%d) does not exist.', $accountId));
}
$this->defaultAccount = $result->getAccount();
RequestCache::set($cacheKey, $token, $this->defaultAccount);
}
}
/**
* @throws ImporterErrorException
*/
private function getDefaultCurrency(): void
{
$url = SecretManager::getBaseUrl();
$token = SecretManager::getAccessToken();
$currencyRequest = new GetCurrencyRequest($url, $token);
$currencyRequest->setVerify(config('importer.connection.verify'));
$currencyRequest->setTimeOut(config('importer.connection.timeout'));
$currencyRequest->setCode('default');
try {
/** @var GetCurrencyResponse $result */
$result = $currencyRequest->get();
$this->defaultCurrency = $result->getCurrency();
} catch (ApiHttpException $e) {
Log::error(sprintf('[%s]: %s', config('importer.version'), $e->getMessage()));
throw new ImporterErrorException('The default currency could not be loaded.');
}
Log::debug(sprintf('Currency found, default currency is assumed to be "%s" (#%d)', $this->defaultCurrency->code, $this->defaultCurrency->id));
}
public function processPseudo(array $lines): array
{
Log::debug(sprintf('[%s] Now in %s', config('importer.version'), __METHOD__));
$count = count($lines);
$processed = [];
Log::info(sprintf('Converting %d line(s) into transactions.', $count));
/** @var array $line */
foreach ($lines as $index => $line) {
Log::info(sprintf('Now processing line %d/%d.', $index + 1, $count));
$processed[] = $this->processPseudoLine($line);
// $this->addMessage($index, sprintf('Converted CSV line %d into a transaction.', $index + 1));
}
Log::info(sprintf('Done converting %d line(s) into transactions.', $count));
return $processed;
}
private function processPseudoLine(array $line): array
{
Log::debug(sprintf('[%s] Now in %s', config('importer.version'), __METHOD__));
foreach ($this->tasks as $task) {
/** @var AbstractTask $object */
$object = app($task);
Log::debug(sprintf('Now running task %s', $task));
if ($object->requiresDefaultAccount()) {
$object->setAccount($this->defaultAccount);
}
if ($object->requiresTransactionCurrency()) {
$object->setTransactionCurrency($this->defaultCurrency);
}
$line = $object->process($line);
}
Log::debug('Final transaction: ', $line);
return $line;
}
}