2017-08-18 07:45:42 -05:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* ExpandedProcessor.php
|
|
|
|
* Copyright (c) 2017 thegrumpydictator@gmail.com
|
|
|
|
*
|
2017-10-21 01:40:00 -05:00
|
|
|
* This file is part of Firefly III.
|
|
|
|
*
|
|
|
|
* Firefly III is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* Firefly III 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 General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with Firefly III. If not, see <http://www.gnu.org/licenses/>.
|
2017-08-18 07:45:42 -05:00
|
|
|
*/
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace FireflyIII\Export;
|
|
|
|
|
|
|
|
use Crypt;
|
|
|
|
use DB;
|
|
|
|
use FireflyIII\Exceptions\FireflyException;
|
|
|
|
use FireflyIII\Export\Collector\AttachmentCollector;
|
|
|
|
use FireflyIII\Export\Collector\UploadCollector;
|
|
|
|
use FireflyIII\Export\Entry\Entry;
|
|
|
|
use FireflyIII\Helpers\Collector\JournalCollectorInterface;
|
|
|
|
use FireflyIII\Helpers\Filter\InternalTransferFilter;
|
|
|
|
use FireflyIII\Models\AccountMeta;
|
|
|
|
use FireflyIII\Models\ExportJob;
|
|
|
|
use FireflyIII\Models\Transaction;
|
|
|
|
use FireflyIII\Models\TransactionJournalMeta;
|
2017-08-18 08:14:44 -05:00
|
|
|
use FireflyIII\Repositories\Currency\CurrencyRepositoryInterface;
|
2017-08-18 07:45:42 -05:00
|
|
|
use Illuminate\Support\Collection;
|
|
|
|
use Log;
|
|
|
|
use Storage;
|
|
|
|
use ZipArchive;
|
|
|
|
|
|
|
|
/**
|
2017-11-15 05:25:49 -06:00
|
|
|
* Class ExpandedProcessor.
|
2017-08-18 07:45:42 -05:00
|
|
|
*
|
2017-09-16 00:41:03 -05:00
|
|
|
* @SuppressWarnings(PHPMD.CouplingBetweenObjects) // its doing a lot.
|
2017-08-18 07:45:42 -05:00
|
|
|
*/
|
|
|
|
class ExpandedProcessor implements ProcessorInterface
|
|
|
|
{
|
|
|
|
/** @var Collection */
|
|
|
|
public $accounts;
|
2017-11-15 05:25:49 -06:00
|
|
|
/** @var string */
|
2017-08-18 07:45:42 -05:00
|
|
|
public $exportFormat;
|
2017-11-15 05:25:49 -06:00
|
|
|
/** @var bool */
|
2017-08-18 07:45:42 -05:00
|
|
|
public $includeAttachments;
|
2017-11-15 05:25:49 -06:00
|
|
|
/** @var bool */
|
2017-08-18 07:45:42 -05:00
|
|
|
public $includeOldUploads;
|
2017-11-15 05:25:49 -06:00
|
|
|
/** @var ExportJob */
|
2017-08-18 07:45:42 -05:00
|
|
|
public $job;
|
|
|
|
/** @var array */
|
|
|
|
public $settings;
|
2017-11-15 05:25:49 -06:00
|
|
|
/** @var Collection */
|
2017-08-18 07:45:42 -05:00
|
|
|
private $exportEntries;
|
2017-11-15 05:25:49 -06:00
|
|
|
/** @var Collection */
|
2017-08-18 07:45:42 -05:00
|
|
|
private $files;
|
2017-11-15 05:25:49 -06:00
|
|
|
/** @var Collection */
|
2017-08-18 07:45:42 -05:00
|
|
|
private $journals;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Processor constructor.
|
|
|
|
*/
|
|
|
|
public function __construct()
|
|
|
|
{
|
|
|
|
$this->journals = new Collection;
|
|
|
|
$this->exportEntries = new Collection;
|
|
|
|
$this->files = new Collection;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function collectAttachments(): bool
|
|
|
|
{
|
|
|
|
/** @var AttachmentCollector $attachmentCollector */
|
|
|
|
$attachmentCollector = app(AttachmentCollector::class);
|
|
|
|
$attachmentCollector->setJob($this->job);
|
|
|
|
$attachmentCollector->setDates($this->settings['startDate'], $this->settings['endDate']);
|
|
|
|
$attachmentCollector->run();
|
|
|
|
$this->files = $this->files->merge($attachmentCollector->getEntries());
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-09-16 00:41:03 -05:00
|
|
|
* Collects all transaction journals.
|
|
|
|
*
|
2017-08-18 07:45:42 -05:00
|
|
|
* @return bool
|
2017-09-16 00:41:03 -05:00
|
|
|
* @SuppressWarnings(PHPMD.ExcessiveMethodLength)
|
2017-08-18 07:45:42 -05:00
|
|
|
*/
|
|
|
|
public function collectJournals(): bool
|
|
|
|
{
|
|
|
|
// use journal collector thing.
|
|
|
|
/** @var JournalCollectorInterface $collector */
|
|
|
|
$collector = app(JournalCollectorInterface::class);
|
2017-09-14 14:17:19 -05:00
|
|
|
$collector->setUser($this->job->user);
|
2017-08-18 07:45:42 -05:00
|
|
|
$collector->setAccounts($this->accounts)->setRange($this->settings['startDate'], $this->settings['endDate'])
|
|
|
|
->withOpposingAccount()->withBudgetInformation()->withCategoryInformation()
|
|
|
|
->removeFilter(InternalTransferFilter::class);
|
|
|
|
$transactions = $collector->getJournals();
|
|
|
|
// get some more meta data for each entry:
|
|
|
|
$ids = $transactions->pluck('journal_id')->toArray();
|
|
|
|
$assetIds = $transactions->pluck('account_id')->toArray();
|
|
|
|
$opposingIds = $transactions->pluck('opposing_account_id')->toArray();
|
|
|
|
$notes = $this->getNotes($ids);
|
|
|
|
$tags = $this->getTags($ids);
|
2017-09-16 00:41:03 -05:00
|
|
|
/** @var array $ibans */
|
2017-10-05 04:49:06 -05:00
|
|
|
$ibans = $this->getIbans($assetIds) + $this->getIbans($opposingIds);
|
|
|
|
$currencies = $this->getAccountCurrencies($ibans);
|
2017-08-18 07:45:42 -05:00
|
|
|
$transactions->each(
|
2017-08-18 08:14:44 -05:00
|
|
|
function (Transaction $transaction) use ($notes, $tags, $ibans, $currencies) {
|
2017-08-18 07:45:42 -05:00
|
|
|
$journalId = intval($transaction->journal_id);
|
|
|
|
$accountId = intval($transaction->account_id);
|
|
|
|
$opposingId = intval($transaction->opposing_account_id);
|
2017-08-18 08:14:44 -05:00
|
|
|
$currencyId = $ibans[$accountId]['currency_id'] ?? 0;
|
|
|
|
$opposingCurrencyId = $ibans[$opposingId]['currency_id'] ?? 0;
|
2017-08-18 07:45:42 -05:00
|
|
|
$transaction->notes = $notes[$journalId] ?? '';
|
|
|
|
$transaction->tags = join(',', $tags[$journalId] ?? []);
|
|
|
|
$transaction->account_number = $ibans[$accountId]['accountNumber'] ?? '';
|
|
|
|
$transaction->account_bic = $ibans[$accountId]['BIC'] ?? '';
|
2017-08-18 08:14:44 -05:00
|
|
|
$transaction->account_currency_code = $currencies[$currencyId] ?? '';
|
2017-08-18 07:45:42 -05:00
|
|
|
$transaction->opposing_account_number = $ibans[$opposingId]['accountNumber'] ?? '';
|
|
|
|
$transaction->opposing_account_bic = $ibans[$opposingId]['BIC'] ?? '';
|
2017-08-18 08:14:44 -05:00
|
|
|
$transaction->opposing_currency_code = $currencies[$opposingCurrencyId] ?? '';
|
2017-08-18 07:45:42 -05:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->journals = $transactions;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function collectOldUploads(): bool
|
|
|
|
{
|
|
|
|
/** @var UploadCollector $uploadCollector */
|
|
|
|
$uploadCollector = app(UploadCollector::class);
|
|
|
|
$uploadCollector->setJob($this->job);
|
|
|
|
$uploadCollector->run();
|
|
|
|
|
|
|
|
$this->files = $this->files->merge($uploadCollector->getEntries());
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function convertJournals(): bool
|
|
|
|
{
|
|
|
|
$this->journals->each(
|
|
|
|
function (Transaction $transaction) {
|
|
|
|
$this->exportEntries->push(Entry::fromTransaction($transaction));
|
|
|
|
}
|
|
|
|
);
|
|
|
|
Log::debug(sprintf('Count %d entries in exportEntries (convertJournals)', $this->exportEntries->count()));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return bool
|
2017-11-15 05:25:49 -06:00
|
|
|
*
|
2017-08-18 07:45:42 -05:00
|
|
|
* @throws FireflyException
|
|
|
|
*/
|
|
|
|
public function createZipFile(): bool
|
|
|
|
{
|
|
|
|
$zip = new ZipArchive;
|
|
|
|
$file = $this->job->key . '.zip';
|
|
|
|
$fullPath = storage_path('export') . '/' . $file;
|
|
|
|
|
2017-11-15 05:25:49 -06:00
|
|
|
if (true !== $zip->open($fullPath, ZipArchive::CREATE)) {
|
2017-08-18 07:45:42 -05:00
|
|
|
throw new FireflyException('Cannot store zip file.');
|
|
|
|
}
|
|
|
|
// for each file in the collection, add it to the zip file.
|
|
|
|
$disk = Storage::disk('export');
|
|
|
|
foreach ($this->getFiles() as $entry) {
|
|
|
|
// is part of this job?
|
|
|
|
$zipFileName = str_replace($this->job->key . '-', '', $entry);
|
|
|
|
$zip->addFromString($zipFileName, $disk->get($entry));
|
|
|
|
}
|
|
|
|
|
|
|
|
$zip->close();
|
|
|
|
|
|
|
|
// delete the files:
|
|
|
|
$this->deleteFiles();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function exportJournals(): bool
|
|
|
|
{
|
|
|
|
$exporterClass = config('firefly.export_formats.' . $this->exportFormat);
|
|
|
|
$exporter = app($exporterClass);
|
|
|
|
$exporter->setJob($this->job);
|
|
|
|
$exporter->setEntries($this->exportEntries);
|
|
|
|
$exporter->run();
|
|
|
|
$this->files->push($exporter->getFileName());
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Collection
|
|
|
|
*/
|
|
|
|
public function getFiles(): Collection
|
|
|
|
{
|
|
|
|
return $this->files;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save export job settings to class.
|
|
|
|
*
|
|
|
|
* @param array $settings
|
|
|
|
*/
|
|
|
|
public function setSettings(array $settings)
|
|
|
|
{
|
|
|
|
// save settings
|
|
|
|
$this->settings = $settings;
|
|
|
|
$this->accounts = $settings['accounts'];
|
|
|
|
$this->exportFormat = $settings['exportFormat'];
|
|
|
|
$this->includeAttachments = $settings['includeAttachments'];
|
|
|
|
$this->includeOldUploads = $settings['includeOldUploads'];
|
|
|
|
$this->job = $settings['job'];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
private function deleteFiles()
|
|
|
|
{
|
|
|
|
$disk = Storage::disk('export');
|
|
|
|
foreach ($this->getFiles() as $file) {
|
|
|
|
$disk->delete($file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-18 08:14:44 -05:00
|
|
|
/**
|
|
|
|
* @param array $array
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
private function getAccountCurrencies(array $array): array
|
|
|
|
{
|
|
|
|
/** @var CurrencyRepositoryInterface $repository */
|
|
|
|
$repository = app(CurrencyRepositoryInterface::class);
|
|
|
|
$return = [];
|
|
|
|
$ids = [];
|
|
|
|
$repository->setUser($this->job->user);
|
|
|
|
foreach ($array as $value) {
|
|
|
|
$ids[] = $value['currency_id'] ?? 0;
|
|
|
|
}
|
|
|
|
$ids = array_unique($ids);
|
|
|
|
$result = $repository->getByIds($ids);
|
|
|
|
|
|
|
|
foreach ($result as $currency) {
|
|
|
|
$return[$currency->id] = $currency->code;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $return;
|
|
|
|
}
|
|
|
|
|
2017-08-18 07:45:42 -05:00
|
|
|
/**
|
2017-11-15 05:25:49 -06:00
|
|
|
* Get all IBAN / SWIFT / account numbers.
|
2017-08-18 07:45:42 -05:00
|
|
|
*
|
|
|
|
* @param array $array
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
private function getIbans(array $array): array
|
|
|
|
{
|
|
|
|
$array = array_unique($array);
|
|
|
|
$return = [];
|
|
|
|
$set = AccountMeta::whereIn('account_id', $array)
|
|
|
|
->leftJoin('accounts', 'accounts.id', 'account_meta.account_id')
|
|
|
|
->where('accounts.user_id', $this->job->user_id)
|
|
|
|
->whereIn('account_meta.name', ['accountNumber', 'BIC', 'currency_id'])
|
|
|
|
->get(['account_meta.id', 'account_meta.account_id', 'account_meta.name', 'account_meta.data']);
|
|
|
|
/** @var AccountMeta $meta */
|
|
|
|
foreach ($set as $meta) {
|
|
|
|
$id = intval($meta->account_id);
|
|
|
|
$return[$id][$meta->name] = $meta->data;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns, if present, for the given journal ID's the notes.
|
|
|
|
*
|
|
|
|
* @param array $array
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
private function getNotes(array $array): array
|
|
|
|
{
|
|
|
|
$array = array_unique($array);
|
|
|
|
$set = TransactionJournalMeta::whereIn('journal_meta.transaction_journal_id', $array)
|
|
|
|
->leftJoin('transaction_journals', 'transaction_journals.id', '=', 'journal_meta.transaction_journal_id')
|
|
|
|
->where('transaction_journals.user_id', $this->job->user_id)
|
|
|
|
->where('journal_meta.name', 'notes')->get(
|
|
|
|
['journal_meta.transaction_journal_id', 'journal_meta.data', 'journal_meta.id']
|
|
|
|
);
|
|
|
|
$return = [];
|
|
|
|
/** @var TransactionJournalMeta $meta */
|
|
|
|
foreach ($set as $meta) {
|
|
|
|
$id = intval($meta->transaction_journal_id);
|
|
|
|
$return[$id] = $meta->data;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a comma joined list of all the users tags linked to these journals.
|
|
|
|
*
|
|
|
|
* @param array $array
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
private function getTags(array $array): array
|
|
|
|
{
|
|
|
|
$set = DB::table('tag_transaction_journal')
|
|
|
|
->whereIn('tag_transaction_journal.transaction_journal_id', $array)
|
|
|
|
->leftJoin('tags', 'tag_transaction_journal.tag_id', '=', 'tags.id')
|
|
|
|
->leftJoin('transaction_journals', 'transaction_journals.id', '=', 'tag_transaction_journal.transaction_journal_id')
|
|
|
|
->where('transaction_journals.user_id', $this->job->user_id)
|
|
|
|
->get(['tag_transaction_journal.transaction_journal_id', 'tags.tag']);
|
|
|
|
$result = [];
|
|
|
|
foreach ($set as $entry) {
|
|
|
|
$id = intval($entry->transaction_journal_id);
|
|
|
|
$result[$id] = isset($result[$id]) ? $result[$id] : [];
|
|
|
|
$result[$id][] = Crypt::decrypt($entry->tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
2017-08-30 23:47:18 -05:00
|
|
|
}
|