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.
 
 
 
 
 

262 lines
10 KiB

<?php
/**
* GenerateTransactions.php
* Copyright (c) 2020 james@firefly-iii.org
*
* This file is part of the Firefly III Spectre importer
* (https://github.com/firefly-iii/spectre-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\Spectre\Conversion\Routine;
use App\Services\Shared\Configuration\Configuration;
use App\Services\Shared\Conversion\ProgressInformation;
use App\Services\Spectre\Model\Transaction;
use App\Support\Http\CollectsAccounts;
use App\Support\Internal\DuplicateSafetyCatch;
use GrumpyDictator\FFIIIApiSupport\Exceptions\ApiHttpException;
/**
* Class GenerateTransactions.
*/
class GenerateTransactions
{
use ProgressInformation;
use CollectsAccounts;
use DuplicateSafetyCatch;
private array $accounts;
private Configuration $configuration;
private array $specialSubTypes = ['REVERSAL', 'REQUEST', 'BILLING', 'SCT', 'SDD', 'NLO'];
private array $targetAccounts;
private array $targetTypes;
/**
* GenerateTransactions constructor.
*/
public function __construct()
{
$this->targetAccounts = [];
$this->targetTypes = [];
bcscale(12);
}
/**
*
* @throws ApiHttpException
*/
public function collectTargetAccounts(): void
{
app('log')->debug('Spectre: Defer account search to trait.');
// defer to trait:
$array = $this->collectAllTargetAccounts();
foreach ($array as $number => $info) {
$this->targetAccounts[$number] = $info['id'];
$this->targetTypes[$number] = $info['type'];
}
app('log')->debug(sprintf('Spectre: Collected %d accounts.', count($this->targetAccounts)));
}
/**
* @param array $spectre
*
* @return array
*/
public function getTransactions(array $spectre): array
{
$return = [];
/** @var Transaction $entry */
foreach ($spectre as $entry) {
$return[] = $this->generateTransaction($entry);
// TODO error handling at this point.
}
//$this->addMessage(0, sprintf('Parsed %d Spectre transactions for further processing.', count($return)));
return $return;
}
/**
* @param Transaction $entry
*
* @return array
*/
private function generateTransaction(Transaction $entry): array
{
app('log')->debug('Original Spectre transaction', $entry->toArray());
$description = $entry->getDescription();
$spectreAccountId = $entry->getAccountId();
$madeOn = $entry->getMadeOn()->toW3cString();
$amount = $entry->getAmount();
// extra information from the "extra" array. May be NULL.
$notes = trim(sprintf('%s %s', $entry->extra->getInformation(), $entry->extra->getAdditional()));
$transaction = [
'type' => 'withdrawal', // reverse
'date' => str_replace('T', ' ', substr($madeOn, 0, 19)),
'datetime' => $madeOn, // not used in API, only for transaction filtering.
'amount' => 0,
'description' => $description,
'order' => 0,
'currency_code' => $entry->getCurrencyCode(),
'tags' => [$entry->getMode(), $entry->getStatus(), $entry->getCategory()],
'category_name' => $entry->getCategory(),
'category_id' => $this->configuration->getMapping()['categories'][$entry->getCategory()] ?? null,
'external_id' => $entry->getId(),
'interal_reference' => $entry->getAccountId(),
'notes' => $notes,
];
$return = [
'apply_rules' => $this->configuration->isRules(),
'error_if_duplicate_hash' => $this->configuration->isIgnoreDuplicateTransactions(),
'transactions' => [],
];
if ($this->configuration->isIgnoreSpectreCategories()) {
app('log')->debug('Remove Spectre categories and tags.');
unset($transaction['tags']);
unset($transaction['category_name']);
unset($transaction['category_id']);
}
// amount is positive?
if (1 === bccomp($amount, '0')) {
app('log')->debug('Amount is positive: assume transfer or deposit.');
$transaction = $this->processPositiveTransaction($entry, $transaction, $amount, $spectreAccountId);
}
if (-1 === bccomp($amount, '0')) {
app('log')->debug('Amount is negative: assume transfer or withdrawal.');
$transaction = $this->processNegativeTransaction($entry, $transaction, $amount, $spectreAccountId);
}
$return['transactions'][] = $transaction;
app('log')->debug(sprintf('Parsed Spectre transaction #%d', $entry->getId()));
return $return;
}
/**
* @param Transaction $entry
* @param array $transaction
* @param string $amount
* @param string $spectreAccountId
*
* @return array
*/
private function processPositiveTransaction(Transaction $entry, array $transaction, string $amount, string $spectreAccountId): array
{
// amount is positive: deposit or transfer. Spectre account is destination
$transaction['type'] = 'deposit';
$transaction['amount'] = $amount;
// destination is Spectre
$transaction['destination_id'] = (int)$this->accounts[$spectreAccountId];
// source is the other side (name!)
// payee is the destination, payer is the source.
// since we know the destination already, we're looking for the payer here:
$transaction['source_name'] = $entry->getPayer() ?? '(unknown source account)';
$transaction['source_iban'] = $entry->getPayerIban() ?? '';
app('log')->debug(sprintf('processPositiveTransaction: source_name = "%s", source_iban = "%s"', $transaction['source_name'], $transaction['source_iban']));
// check if the source IBAN is a known account and what type it has: perhaps the
// transaction type needs to be changed:
$iban = $transaction['source_iban'];
$accountType = $this->targetTypes[$iban] ?? 'unknown';
$accountId = $this->targetAccounts[$iban] ?? 0;
app('log')->debug(sprintf('Found account type "%s" for IBAN "%s"', $accountType, $iban));
if ('unknown' !== $accountType) {
if ('asset' === $accountType) {
app('log')->debug('Changing transaction type to "transfer"');
$transaction['type'] = 'transfer';
}
}
if (0 !== $accountId) {
app('log')->debug(sprintf('Found account ID #%d for IBAN "%s"', $accountId, $iban));
$transaction['source_id'] = (int)$accountId;
unset($transaction['source_name'], $transaction['source_iban']);
}
$transaction = $this->positiveTransactionSafetyCatch($transaction, $entry->getPayer(), $entry->getPayerIban());
app('log')->debug(sprintf('destination_id = %d, source_name = "%s", source_iban = "%s", source_id = "%s"', $transaction['destination_id'] ?? '', $transaction['source_name'] ?? '', $transaction['source_iban'] ?? '', $transaction['source_id'] ?? ''));
return $transaction;
}
/**
* @param Transaction $entry
* @param array $transaction
* @param string $amount
* @param string $spectreAccountId
*
* @return array
*/
private function processNegativeTransaction(Transaction $entry, array $transaction, string $amount, string $spectreAccountId): array
{
// amount is negative: withdrawal or transfer.
$transaction['amount'] = bcmul($amount, '-1');
// source is Spectre:
$transaction['source_id'] = (int)$this->accounts[$spectreAccountId];
// dest is shop. Payee / payer is reverse from the other one.
$transaction['destination_name'] = $entry->getPayee() ?? '(unknown destination account)';
$transaction['destination_iban'] = $entry->getPayeeIban() ?? '';
app('log')->debug(sprintf('processNegativeTransaction: destination_name = "%s", destination_iban = "%s"', $transaction['destination_name'], $transaction['destination_iban']));
// check if the destination IBAN is a known account and what type it has: perhaps the
// transaction type needs to be changed:
$iban = $transaction['destination_iban'];
$accountType = $this->targetTypes[$iban] ?? 'unknown';
$accountId = $this->targetAccounts[$iban] ?? 0;
app('log')->debug(sprintf('Found account type "%s" for IBAN "%s"', $accountType, $iban));
if ('unknown' !== $accountType) {
if ('asset' === $accountType) {
app('log')->debug('Changing transaction type to "transfer"');
$transaction['type'] = 'transfer';
}
}
if (0 !== $accountId) {
app('log')->debug(sprintf('Found account ID #%d for IBAN "%s"', $accountId, $iban));
$transaction['destination_id'] = $accountId;
unset($transaction['destination_name'], $transaction['destination_iban']);
}
$transaction = $this->negativeTransactionSafetyCatch($transaction, $entry->getPayee(), $entry->getPayeeIban());
app('log')->debug(sprintf('source_id = %d, destination_id = "%s", destination_name = "%s", destination_iban = "%s"', $transaction['source_id'], $transaction['destination_id'] ?? '', $transaction['destination_name'] ?? '', $transaction['destination_iban'] ?? ''));
return $transaction;
}
/**
* @param Configuration $configuration
*/
public function setConfiguration(Configuration $configuration): void
{
$this->configuration = $configuration;
$this->accounts = $configuration->getAccounts();
}
}