mirror of
https://github.com/firefly-iii/firefly-iii.git
synced 2025-10-24 04:41:01 +00:00
371 lines
17 KiB
PHP
371 lines
17 KiB
PHP
<?php
|
|
/**
|
|
* BillTransformer.php
|
|
* Copyright (c) 2019 james@firefly-iii.org
|
|
*
|
|
* This file is part of Firefly III (https://github.com/firefly-iii).
|
|
*
|
|
* 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 FireflyIII\Transformers\V2;
|
|
|
|
use Carbon\Carbon;
|
|
use Carbon\CarbonInterface;
|
|
use FireflyIII\Exceptions\FireflyException;
|
|
use FireflyIII\Models\Bill;
|
|
use FireflyIII\Models\Note;
|
|
use FireflyIII\Models\ObjectGroup;
|
|
use FireflyIII\Models\Transaction;
|
|
use FireflyIII\Models\TransactionCurrency;
|
|
use FireflyIII\Models\TransactionJournal;
|
|
use FireflyIII\Support\Http\Api\ExchangeRateConverter;
|
|
use Illuminate\Support\Collection;
|
|
use Illuminate\Support\Facades\DB;
|
|
use Illuminate\Support\Facades\Log;
|
|
|
|
/**
|
|
* Class BillTransformer
|
|
*/
|
|
class BillTransformer extends AbstractTransformer
|
|
{
|
|
private ExchangeRateConverter $converter;
|
|
private array $currencies;
|
|
private TransactionCurrency $default;
|
|
private array $groups;
|
|
private array $notes;
|
|
private array $paidDates;
|
|
|
|
/**
|
|
* @throws FireflyException
|
|
*
|
|
* @SuppressWarnings(PHPMD.ExcessiveMethodLength)
|
|
*/
|
|
public function collectMetaData(Collection $objects): Collection
|
|
{
|
|
$currencies = [];
|
|
$bills = [];
|
|
$this->notes = [];
|
|
$this->groups = [];
|
|
$this->paidDates = [];
|
|
|
|
/** @var Bill $object */
|
|
foreach ($objects as $object) {
|
|
$id = $object->transaction_currency_id;
|
|
$bills[] = $object->id;
|
|
$currencies[$id] ??= TransactionCurrency::find($id);
|
|
}
|
|
$this->currencies = $currencies;
|
|
$notes = Note::whereNoteableType(Bill::class)->whereIn('noteable_id', array_keys($bills))->get();
|
|
|
|
/** @var Note $note */
|
|
foreach ($notes as $note) {
|
|
$id = $note->noteable_id;
|
|
$this->notes[$id] = $note;
|
|
}
|
|
// grab object groups:
|
|
$set = DB::table('object_groupables')
|
|
->leftJoin('object_groups', 'object_groups.id', '=', 'object_groupables.object_group_id')
|
|
->where('object_groupables.object_groupable_type', Bill::class)
|
|
->get(['object_groupables.*', 'object_groups.title', 'object_groups.order'])
|
|
;
|
|
|
|
/** @var ObjectGroup $entry */
|
|
foreach ($set as $entry) {
|
|
$billId = (int)$entry->object_groupable_id;
|
|
$id = (int)$entry->object_group_id;
|
|
$order = $entry->order;
|
|
$this->groups[$billId] = [
|
|
'object_group_id' => $id,
|
|
'object_group_title' => $entry->title,
|
|
'object_group_order' => $order,
|
|
];
|
|
}
|
|
Log::debug(sprintf('Created new ExchangeRateConverter in %s', __METHOD__));
|
|
$this->default = app('amount')->getDefaultCurrency();
|
|
$this->converter = new ExchangeRateConverter();
|
|
|
|
// grab all paid dates:
|
|
if (null !== $this->parameters->get('start') && null !== $this->parameters->get('end')) {
|
|
$journals = TransactionJournal::whereIn('bill_id', $bills)
|
|
->where('date', '>=', $this->parameters->get('start'))
|
|
->where('date', '<=', $this->parameters->get('end'))
|
|
->get(['transaction_journals.id', 'transaction_journals.transaction_group_id', 'transaction_journals.date', 'transaction_journals.bill_id'])
|
|
;
|
|
$journalIds = $journals->pluck('id')->toArray();
|
|
|
|
// grab transactions for amount:
|
|
$set = Transaction::whereIn('transaction_journal_id', $journalIds)
|
|
->where('transactions.amount', '<', 0)
|
|
->get(['transactions.id', 'transactions.transaction_journal_id', 'transactions.amount', 'transactions.foreign_amount', 'transactions.transaction_currency_id', 'transactions.foreign_currency_id'])
|
|
;
|
|
$transactions = [];
|
|
|
|
/** @var Transaction $transaction */
|
|
foreach ($set as $transaction) {
|
|
$journalId = $transaction->transaction_journal_id;
|
|
$transactions[$journalId] = $transaction->toArray();
|
|
}
|
|
|
|
/** @var TransactionJournal $journal */
|
|
foreach ($journals as $journal) {
|
|
app('log')->debug(sprintf('Processing journal #%d', $journal->id));
|
|
$transaction = $transactions[$journal->id] ?? [];
|
|
$billId = (int)$journal->bill_id;
|
|
$currencyId = (int)($transaction['transaction_currency_id'] ?? 0);
|
|
$currencies[$currencyId] ??= TransactionCurrency::find($currencyId);
|
|
|
|
// foreign currency
|
|
$foreignCurrencyId = null;
|
|
$foreignCurrencyCode = null;
|
|
$foreignCurrencyName = null;
|
|
$foreignCurrencySymbol = null;
|
|
$foreignCurrencyDp = null;
|
|
app('log')->debug('Foreign currency is NULL');
|
|
if (null !== $transaction['foreign_currency_id']) {
|
|
app('log')->debug(sprintf('Foreign currency is #%d', $transaction['foreign_currency_id']));
|
|
$foreignCurrencyId = (int)$transaction['foreign_currency_id'];
|
|
$currencies[$foreignCurrencyId] ??= TransactionCurrency::find($foreignCurrencyId);
|
|
$foreignCurrencyCode = $currencies[$foreignCurrencyId]->code;
|
|
$foreignCurrencyName = $currencies[$foreignCurrencyId]->name;
|
|
$foreignCurrencySymbol = $currencies[$foreignCurrencyId]->symbol;
|
|
$foreignCurrencyDp = $currencies[$foreignCurrencyId]->decimal_places;
|
|
}
|
|
|
|
$this->paidDates[$billId][] = [
|
|
'transaction_group_id' => (string)$journal->id,
|
|
'transaction_journal_id' => (string)$journal->transaction_group_id,
|
|
'date' => $journal->date->toAtomString(),
|
|
'currency_id' => $currencies[$currencyId]->id,
|
|
'currency_code' => $currencies[$currencyId]->code,
|
|
'currency_name' => $currencies[$currencyId]->name,
|
|
'currency_symbol' => $currencies[$currencyId]->symbol,
|
|
'currency_decimal_places' => $currencies[$currencyId]->decimal_places,
|
|
'native_currency_id' => $currencies[$currencyId]->id,
|
|
'native_currency_code' => $currencies[$currencyId]->code,
|
|
'native_currency_symbol' => $currencies[$currencyId]->symbol,
|
|
'native_currency_decimal_places' => $currencies[$currencyId]->decimal_places,
|
|
'foreign_currency_id' => $foreignCurrencyId,
|
|
'foreign_currency_code' => $foreignCurrencyCode,
|
|
'foreign_currency_name' => $foreignCurrencyName,
|
|
'foreign_currency_symbol' => $foreignCurrencySymbol,
|
|
'foreign_currency_decimal_places' => $foreignCurrencyDp,
|
|
'amount' => $transaction['amount'],
|
|
'foreign_amount' => $transaction['foreign_amount'],
|
|
'native_amount' => $this->converter->convert($currencies[$currencyId], $this->default, $journal->date, $transaction['amount']),
|
|
'foreign_native_amount' => '' === (string)$transaction['foreign_amount'] ? null : $this->converter->convert(
|
|
$currencies[$foreignCurrencyId],
|
|
$this->default,
|
|
$journal->date,
|
|
$transaction['foreign_amount']
|
|
),
|
|
];
|
|
}
|
|
}
|
|
|
|
return $objects;
|
|
}
|
|
|
|
/**
|
|
* Transform the bill.
|
|
*/
|
|
public function transform(Bill $bill): array
|
|
{
|
|
$paidData = $this->paidDates[$bill->id] ?? [];
|
|
$nextExpectedMatch = $this->nextExpectedMatch($bill, $this->paidDates[$bill->id] ?? []);
|
|
$payDates = $this->payDates($bill);
|
|
$currency = $this->currencies[$bill->transaction_currency_id];
|
|
$group = $this->groups[$bill->id] ?? null;
|
|
|
|
// date for currency conversion
|
|
/** @var null|Carbon $startParam */
|
|
$startParam = $this->parameters->get('start');
|
|
|
|
/** @var null|Carbon $date */
|
|
$date = null === $startParam ? today() : clone $startParam;
|
|
|
|
$nextExpectedMatchDiff = $this->getNextExpectedMatchDiff($nextExpectedMatch, $payDates);
|
|
$this->converter->summarize();
|
|
|
|
return [
|
|
'id' => $bill->id,
|
|
'created_at' => $bill->created_at->toAtomString(),
|
|
'updated_at' => $bill->updated_at->toAtomString(),
|
|
'name' => $bill->name,
|
|
'amount_min' => app('steam')->bcround($bill->amount_min, $currency->decimal_places),
|
|
'amount_max' => app('steam')->bcround($bill->amount_max, $currency->decimal_places),
|
|
'native_amount_min' => $this->converter->convert($currency, $this->default, $date, $bill->amount_min),
|
|
'native_amount_max' => $this->converter->convert($currency, $this->default, $date, $bill->amount_max),
|
|
'currency_id' => (string)$bill->transaction_currency_id,
|
|
'currency_code' => $currency->code,
|
|
'currency_name' => $currency->name,
|
|
'currency_symbol' => $currency->symbol,
|
|
'currency_decimal_places' => $currency->decimal_places,
|
|
'native_currency_id' => $this->default->id,
|
|
'native_currency_code' => $this->default->code,
|
|
'native_currency_name' => $this->default->name,
|
|
'native_currency_symbol' => $this->default->symbol,
|
|
'native_currency_decimal_places' => $this->default->decimal_places,
|
|
'date' => $bill->date->toAtomString(),
|
|
'end_date' => $bill->end_date?->toAtomString(),
|
|
'extension_date' => $bill->extension_date?->toAtomString(),
|
|
'repeat_freq' => $bill->repeat_freq,
|
|
'skip' => $bill->skip,
|
|
'active' => $bill->active,
|
|
'order' => $bill->order,
|
|
'notes' => $this->notes[$bill->id] ?? null,
|
|
'object_group_id' => $group ? $group['object_group_id'] : null,
|
|
'object_group_order' => $group ? $group['object_group_order'] : null,
|
|
'object_group_title' => $group ? $group['object_group_title'] : null,
|
|
'next_expected_match' => $nextExpectedMatch->toAtomString(),
|
|
'next_expected_match_diff' => $nextExpectedMatchDiff,
|
|
'pay_dates' => $payDates,
|
|
'paid_dates' => $paidData,
|
|
'links' => [
|
|
[
|
|
'rel' => 'self',
|
|
'uri' => sprintf('/bills/%d', $bill->id),
|
|
],
|
|
],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* Get the data the bill was paid and predict the next expected match.
|
|
*/
|
|
protected function nextExpectedMatch(Bill $bill, array $dates): Carbon
|
|
{
|
|
// 2023-07-1 sub one day from the start date to fix a possible bug (see #7704)
|
|
// 2023-07-18 this particular date is used to search for the last paid date.
|
|
// 2023-07-18 the cloned $searchDate is used to grab the correct transactions.
|
|
|
|
/** @var null|Carbon $startParam */
|
|
$startParam = $this->parameters->get('start');
|
|
|
|
/** @var null|Carbon $start */
|
|
$start = null === $startParam ? today() : clone $startParam;
|
|
$start->subDay();
|
|
|
|
$lastPaidDate = $this->lastPaidDate($dates, $start);
|
|
$nextMatch = clone $bill->date;
|
|
while ($nextMatch < $lastPaidDate) {
|
|
/*
|
|
* As long as this date is smaller than the last time the bill was paid, keep jumping ahead.
|
|
* For example: 1 jan, 1 feb, etc.
|
|
*/
|
|
$nextMatch = app('navigation')->addPeriod($nextMatch, $bill->repeat_freq, $bill->skip);
|
|
}
|
|
if ($nextMatch->isSameDay($lastPaidDate)) {
|
|
// Add another period because it's the same day as the last paid date.
|
|
$nextMatch = app('navigation')->addPeriod($nextMatch, $bill->repeat_freq, $bill->skip);
|
|
}
|
|
|
|
return $nextMatch;
|
|
}
|
|
|
|
/**
|
|
* Returns the latest date in the set, or start when set is empty.
|
|
*/
|
|
protected function lastPaidDate(array $dates, Carbon $default): Carbon
|
|
{
|
|
if (0 === count($dates)) {
|
|
return $default;
|
|
}
|
|
$latest = $dates[0]['date'];
|
|
|
|
/** @var array $row */
|
|
foreach ($dates as $row) {
|
|
$carbon = new Carbon($row['date']);
|
|
if ($carbon->gte($latest)) {
|
|
$latest = $row['date'];
|
|
}
|
|
}
|
|
|
|
return new Carbon($latest);
|
|
}
|
|
|
|
protected function payDates(Bill $bill): array
|
|
{
|
|
// app('log')->debug(sprintf('Now in payDates() for bill #%d', $bill->id));
|
|
if (null === $this->parameters->get('start') || null === $this->parameters->get('end')) {
|
|
// app('log')->debug('No start or end date, give empty array.');
|
|
|
|
return [];
|
|
}
|
|
$set = new Collection();
|
|
$currentStart = clone $this->parameters->get('start');
|
|
// 2023-06-23 subDay to fix 7655
|
|
$currentStart->subDay();
|
|
$loop = 0;
|
|
while ($currentStart <= $this->parameters->get('end')) {
|
|
$nextExpectedMatch = $this->nextDateMatch($bill, $currentStart);
|
|
// If nextExpectedMatch is after end, we continue:
|
|
if ($nextExpectedMatch > $this->parameters->get('end')) {
|
|
break;
|
|
}
|
|
// add to set
|
|
$set->push(clone $nextExpectedMatch);
|
|
$nextExpectedMatch->addDay();
|
|
$currentStart = clone $nextExpectedMatch;
|
|
++$loop;
|
|
if ($loop > 4) {
|
|
break;
|
|
}
|
|
}
|
|
$simple = $set->map(
|
|
static function (Carbon $date) {
|
|
return $date->toAtomString();
|
|
}
|
|
);
|
|
|
|
return $simple->toArray();
|
|
}
|
|
|
|
/**
|
|
* Given a bill and a date, this method will tell you at which moment this bill expects its next
|
|
* transaction. Whether or not it is there already, is not relevant.
|
|
*
|
|
* TODO this method is bad compared to the v1 one.
|
|
*/
|
|
protected function nextDateMatch(Bill $bill, Carbon $date): Carbon
|
|
{
|
|
// app('log')->debug(sprintf('Now in nextDateMatch(%d, %s)', $bill->id, $date->format('Y-m-d')));
|
|
$start = clone $bill->date;
|
|
// app('log')->debug(sprintf('Bill start date is %s', $start->format('Y-m-d')));
|
|
while ($start < $date) {
|
|
$start = app('navigation')->addPeriod($start, $bill->repeat_freq, $bill->skip);
|
|
}
|
|
|
|
// app('log')->debug(sprintf('End of loop, bill start date is now %s', $start->format('Y-m-d')));
|
|
|
|
return $start;
|
|
}
|
|
|
|
private function getNextExpectedMatchDiff(Carbon $next, array $dates): string
|
|
{
|
|
if ($next->isToday()) {
|
|
return trans('firefly.today');
|
|
}
|
|
$current = $dates[0] ?? null;
|
|
if (null === $current) {
|
|
return trans('firefly.not_expected_period');
|
|
}
|
|
$carbon = new Carbon($current);
|
|
|
|
return $carbon->diffForHumans(today(config('app.timezone')), CarbonInterface::DIFF_RELATIVE_TO_NOW);
|
|
}
|
|
}
|