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.
210 lines
9.0 KiB
210 lines
9.0 KiB
<?php
|
|
|
|
/*
|
|
* CollectsAccounts.php
|
|
* Copyright (c) 2023 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\Support\Internal;
|
|
|
|
use App\Exceptions\AgreementExpiredException;
|
|
use App\Exceptions\ImporterErrorException;
|
|
use App\Exceptions\ImporterHttpException;
|
|
use App\Services\Nordigen\Model\Account as NordigenAccount;
|
|
use App\Services\Nordigen\Request\ListAccountsRequest;
|
|
use App\Services\Nordigen\Services\AccountInformationCollector;
|
|
use App\Services\Nordigen\TokenManager;
|
|
use App\Services\Session\Constants;
|
|
use App\Services\Shared\Authentication\SecretManager;
|
|
use App\Services\Shared\Configuration\Configuration;
|
|
use App\Services\Spectre\Authentication\SecretManager as SpectreSecretManager;
|
|
use App\Services\Spectre\Request\GetAccountsRequest as SpectreGetAccountsRequest;
|
|
use App\Services\Spectre\Response\GetAccountsResponse;
|
|
use GrumpyDictator\FFIIIApiSupport\Exceptions\ApiHttpException;
|
|
use GrumpyDictator\FFIIIApiSupport\Model\Account;
|
|
use GrumpyDictator\FFIIIApiSupport\Request\GetAccountsRequest;
|
|
use GuzzleHttp\Exception\GuzzleException;
|
|
use Illuminate\Support\Facades\Cache;
|
|
use Illuminate\Support\Facades\Log;
|
|
use Exception;
|
|
|
|
trait CollectsAccounts
|
|
{
|
|
/**
|
|
* @throws ApiHttpException
|
|
*/
|
|
protected function getFireflyIIIAccounts(): array
|
|
{
|
|
$accounts = [
|
|
Constants::ASSET_ACCOUNTS => [],
|
|
Constants::LIABILITIES => [],
|
|
];
|
|
$url = null;
|
|
|
|
try {
|
|
$url = SecretManager::getBaseUrl();
|
|
$token = SecretManager::getAccessToken();
|
|
|
|
if ('' === $url || '' === $token) {
|
|
Log::error('CollectsAccounts::getFireflyIIIAccounts - Base URL or Access Token is empty. Cannot fetch accounts.', ['url_empty' => '' === $url, 'token_empty' => '' === $token]);
|
|
|
|
return $accounts; // Return empty accounts if auth details are missing
|
|
}
|
|
|
|
// Fetch ASSET accounts
|
|
Log::debug('CollectsAccounts::getFireflyIIIAccounts - Fetching ASSET accounts from Firefly III.', ['url' => $url]);
|
|
$requestAsset = new GetAccountsRequest($url, $token);
|
|
$requestAsset->setType(GetAccountsRequest::ASSET);
|
|
$requestAsset->setVerify(config('importer.connection.verify'));
|
|
$requestAsset->setTimeOut(config('importer.connection.timeout'));
|
|
$responseAsset = $requestAsset->get();
|
|
|
|
/** @var Account $account */
|
|
foreach ($responseAsset as $account) {
|
|
$accounts[Constants::ASSET_ACCOUNTS][$account->id] = $account;
|
|
}
|
|
Log::debug(sprintf('CollectsAccounts::getFireflyIIIAccounts - Fetched %d asset accounts.', count($accounts[Constants::ASSET_ACCOUNTS])));
|
|
|
|
// Fetch LIABILITY accounts
|
|
// URL and token are likely the same, but re-fetching defensively or if SecretManager has internal state
|
|
$url = SecretManager::getBaseUrl(); // Re-fetch in case of any state change, though unlikely
|
|
$token = SecretManager::getAccessToken();
|
|
|
|
if ('' === $url || '' === $token) { // Check again, though highly unlikely to change if first call succeeded.
|
|
Log::error('CollectsAccounts::getFireflyIIIAccounts - Base URL or Access Token became empty before fetching LIABILITY accounts.');
|
|
|
|
return $accounts; // Return partially filled or empty accounts
|
|
}
|
|
|
|
Log::debug('CollectsAccounts::getFireflyIIIAccounts - Fetching LIABILITY accounts from Firefly III.', ['url' => $url]);
|
|
$requestLiability = new GetAccountsRequest($url, $token);
|
|
$requestLiability->setVerify(config('importer.connection.verify'));
|
|
$requestLiability->setTimeOut(config('importer.connection.timeout'));
|
|
$requestLiability->setType(GetAccountsRequest::LIABILITIES);
|
|
$responseLiability = $requestLiability->get();
|
|
|
|
/** @var Account $account */
|
|
foreach ($responseLiability as $account) {
|
|
$accounts[Constants::LIABILITIES][$account->id] = $account;
|
|
}
|
|
Log::debug(sprintf('CollectsAccounts::getFireflyIIIAccounts - Fetched %d liability accounts.', count($accounts[Constants::LIABILITIES])));
|
|
|
|
} catch (ApiHttpException $e) {
|
|
Log::error('CollectsAccounts::getFireflyIIIAccounts - ApiHttpException while fetching Firefly III accounts.', [
|
|
'message' => $e->getMessage(),
|
|
'code' => $e->getCode(),
|
|
'url' => $url, // Log URL that might have caused issue
|
|
'trace' => $e->getTraceAsString(),
|
|
]);
|
|
// Return the (potentially partially filled) $accounts array so the app doesn't hard crash.
|
|
// The view should handle cases where account lists are empty.
|
|
} catch (Exception $e) {
|
|
Log::error('CollectsAccounts::getFireflyIIIAccounts - Generic Exception while fetching Firefly III accounts.', [
|
|
'message' => $e->getMessage(),
|
|
'code' => $e->getCode(),
|
|
'url' => $url,
|
|
'trace' => $e->getTraceAsString(),
|
|
]);
|
|
}
|
|
Log::debug('CollectsAccounts::getFireflyIIIAccounts - Returning accounts structure.', $accounts);
|
|
|
|
return $accounts;
|
|
}
|
|
|
|
/**
|
|
* List Nordigen accounts with account details, balances, and 2 transactions (if present)
|
|
*
|
|
* @throws AgreementExpiredException|ImporterErrorException
|
|
*/
|
|
protected function getNordigenAccounts(Configuration $configuration): array
|
|
{
|
|
Log::debug(sprintf('Now in %s', __METHOD__));
|
|
$requisitions = $configuration->getNordigenRequisitions();
|
|
$identifier = array_shift($requisitions);
|
|
$inCache = Cache::has($identifier) && config('importer.use_cache');
|
|
$inCache = false;
|
|
|
|
// if cached, return it.
|
|
if ($inCache) {
|
|
$result = Cache::get($identifier);
|
|
$return = [];
|
|
foreach ($result as $arr) {
|
|
$return[] = NordigenAccount::fromLocalArray($arr);
|
|
}
|
|
Log::debug('Grab accounts from cache', $result);
|
|
|
|
return $return;
|
|
}
|
|
// get banks and countries
|
|
$accessToken = TokenManager::getAccessToken();
|
|
$url = config('nordigen.url');
|
|
$request = new ListAccountsRequest($url, $identifier, $accessToken);
|
|
$request->setTimeOut(config('importer.connection.timeout'));
|
|
|
|
// @var ListAccountsResponse $response
|
|
try {
|
|
$response = $request->get();
|
|
} catch (ImporterErrorException|ImporterHttpException $e) {
|
|
throw new ImporterErrorException($e->getMessage(), 0, $e);
|
|
}
|
|
$total = count($response);
|
|
$return = [];
|
|
$cache = [];
|
|
Log::debug(sprintf('Found %d GoCardless accounts.', $total));
|
|
|
|
/** @var NordigenAccount $account */
|
|
foreach ($response as $index => $account) {
|
|
Log::debug(
|
|
sprintf('[%d/%d] Now collecting information for account %s', $index + 1, $total, $account->getIdentifier()),
|
|
$account->toLocalArray()
|
|
);
|
|
$account = AccountInformationCollector::collectInformation($account, true);
|
|
$return[] = $account;
|
|
$cache[] = $account->toLocalArray();
|
|
}
|
|
Cache::put($identifier, $cache, 1800); // half an hour
|
|
|
|
return $return;
|
|
}
|
|
|
|
/**
|
|
* @throws GuzzleException
|
|
* @throws ImporterHttpException
|
|
*/
|
|
protected function getSpectreAccounts(Configuration $configuration): array
|
|
{
|
|
$return = [];
|
|
$url = config('spectre.url');
|
|
$appId = SpectreSecretManager::getAppId();
|
|
$secret = SpectreSecretManager::getSecret();
|
|
$spectreList = new SpectreGetAccountsRequest($url, $appId, $secret);
|
|
$spectreList->setTimeOut(config('importer.connection.timeout'));
|
|
$spectreList->connection = $configuration->getConnection();
|
|
|
|
/** @var GetAccountsResponse $spectreAccounts */
|
|
$spectreAccounts = $spectreList->get();
|
|
foreach ($spectreAccounts as $account) {
|
|
$return[] = $account;
|
|
}
|
|
|
|
return $return;
|
|
}
|
|
}
|