2024-12-21 11:20:48 +01:00
|
|
|
<?php
|
2024-12-21 12:27:07 +01:00
|
|
|
|
|
|
|
declare(strict_types=1);
|
2024-12-21 11:20:48 +01:00
|
|
|
/*
|
|
|
|
* RecalculateNativeAmounts.php
|
|
|
|
* Copyright (c) 2024 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/.
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace FireflyIII\Console\Commands\Correction;
|
|
|
|
|
|
|
|
use FireflyIII\Console\Commands\ShowsFriendlyMessages;
|
2024-12-21 11:33:58 +01:00
|
|
|
use FireflyIII\Handlers\Observer\TransactionObserver;
|
2024-12-21 11:20:48 +01:00
|
|
|
use FireflyIII\Models\Account;
|
|
|
|
use FireflyIII\Models\AutoBudget;
|
|
|
|
use FireflyIII\Models\AvailableBudget;
|
|
|
|
use FireflyIII\Models\Bill;
|
|
|
|
use FireflyIII\Models\Budget;
|
|
|
|
use FireflyIII\Models\BudgetLimit;
|
|
|
|
use FireflyIII\Models\PiggyBank;
|
|
|
|
use FireflyIII\Models\PiggyBankEvent;
|
|
|
|
use FireflyIII\Models\Transaction;
|
|
|
|
use FireflyIII\Models\TransactionCurrency;
|
|
|
|
use FireflyIII\Models\UserGroup;
|
|
|
|
use FireflyIII\Repositories\UserGroup\UserGroupRepositoryInterface;
|
|
|
|
use FireflyIII\Repositories\UserGroups\PiggyBank\PiggyBankRepositoryInterface;
|
2024-12-21 11:33:58 +01:00
|
|
|
use FireflyIII\Support\Facades\Preferences;
|
2024-12-21 11:20:48 +01:00
|
|
|
use FireflyIII\Support\Http\Api\ExchangeRateConverter;
|
|
|
|
use Illuminate\Console\Command;
|
|
|
|
use Illuminate\Database\Eloquent\Builder as EloquentBuilder;
|
|
|
|
use Illuminate\Database\Query\Builder as DatabaseBuilder;
|
|
|
|
use Illuminate\Support\Facades\DB;
|
|
|
|
use Illuminate\Support\Facades\Log;
|
|
|
|
|
|
|
|
class RecalculateNativeAmounts extends Command
|
|
|
|
{
|
|
|
|
use ShowsFriendlyMessages;
|
|
|
|
|
|
|
|
protected $description = 'Recalculate native amounts for all objects.';
|
|
|
|
|
2024-12-21 12:27:07 +01:00
|
|
|
protected $signature = 'firefly-iii:recalculate-native-amounts';
|
2024-12-21 11:20:48 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Execute the console command.
|
|
|
|
*/
|
|
|
|
public function handle(): int
|
|
|
|
{
|
|
|
|
Log::debug('Will update all native amounts. This may take some time.');
|
|
|
|
$this->friendlyWarning('Recalculating native amounts for all objects. This may take some time!');
|
2024-12-21 12:27:07 +01:00
|
|
|
|
2024-12-21 11:20:48 +01:00
|
|
|
/** @var UserGroupRepositoryInterface $repository */
|
|
|
|
$repository = app(UserGroupRepositoryInterface::class);
|
2024-12-21 12:27:07 +01:00
|
|
|
|
2024-12-21 11:20:48 +01:00
|
|
|
/** @var UserGroup $userGroup */
|
|
|
|
foreach ($repository->getAll() as $userGroup) {
|
|
|
|
$this->recalculateForGroup($userGroup);
|
|
|
|
}
|
2024-12-21 11:33:58 +01:00
|
|
|
$this->friendlyInfo('Recalculated all native amounts.');
|
2024-12-21 12:27:07 +01:00
|
|
|
|
2024-12-21 11:20:48 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
private function recalculateForGroup(UserGroup $userGroup): void
|
|
|
|
{
|
|
|
|
Log::debug(sprintf('Now recalculating for user group #%d', $userGroup->id));
|
|
|
|
$this->recalculateAccounts($userGroup);
|
|
|
|
|
|
|
|
// do a check with the group's currency so we can skip some stuff.
|
2024-12-21 11:33:58 +01:00
|
|
|
Preferences::mark();
|
2024-12-21 11:20:48 +01:00
|
|
|
$currency = app('amount')->getDefaultCurrencyByUserGroup($userGroup);
|
|
|
|
|
|
|
|
$this->recalculatePiggyBanks($userGroup, $currency);
|
|
|
|
$this->recalculateBudgets($userGroup, $currency);
|
|
|
|
$this->recalculateAvailableBudgets($userGroup, $currency);
|
|
|
|
$this->recalculateBills($userGroup, $currency);
|
|
|
|
$this->calculateTransactions($userGroup, $currency);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
private function recalculateAccounts(UserGroup $userGroup): void
|
|
|
|
{
|
2024-12-21 12:27:07 +01:00
|
|
|
$set = $userGroup->accounts()->where(function (EloquentBuilder $q): void {
|
2024-12-21 11:20:48 +01:00
|
|
|
$q->whereNotNull('virtual_balance');
|
|
|
|
$q->orWhere('virtual_balance', '!=', '');
|
|
|
|
})->get();
|
2024-12-21 12:27:07 +01:00
|
|
|
|
2024-12-21 11:20:48 +01:00
|
|
|
/** @var Account $account */
|
|
|
|
foreach ($set as $account) {
|
|
|
|
$account->touch();
|
|
|
|
}
|
|
|
|
Log::debug(sprintf('Recalculated %d accounts', $set->count()));
|
|
|
|
}
|
|
|
|
|
|
|
|
private function recalculatePiggyBanks(UserGroup $userGroup, TransactionCurrency $currency): void
|
|
|
|
{
|
2024-12-21 12:27:07 +01:00
|
|
|
$converter = new ExchangeRateConverter();
|
2024-12-21 11:20:48 +01:00
|
|
|
$converter->setIgnoreSettings(true);
|
|
|
|
$repository = app(PiggyBankRepositoryInterface::class);
|
|
|
|
$repository->setUserGroup($userGroup);
|
2024-12-21 12:27:07 +01:00
|
|
|
$set = $repository->getPiggyBanks();
|
|
|
|
$set = $set->filter(
|
2024-12-21 11:20:48 +01:00
|
|
|
static function (PiggyBank $piggyBank) use ($currency) {
|
|
|
|
return $currency->id !== $piggyBank->transaction_currency_id;
|
|
|
|
}
|
|
|
|
);
|
|
|
|
foreach ($set as $piggyBank) {
|
|
|
|
$piggyBank->encrypted = false;
|
|
|
|
$piggyBank->save();
|
|
|
|
|
|
|
|
foreach ($piggyBank->accounts as $account) {
|
|
|
|
$account->pivot->native_current_amount = null;
|
|
|
|
if (0 !== bccomp($account->pivot->current_amount, '0')) {
|
|
|
|
$account->pivot->native_current_amount = $converter->convert($piggyBank->transactionCurrency, $currency, today(), $account->pivot->current_amount);
|
|
|
|
}
|
|
|
|
$account->pivot->save();
|
|
|
|
}
|
|
|
|
$this->recalculatePiggyBankEvents($piggyBank);
|
|
|
|
}
|
|
|
|
Log::debug(sprintf('Recalculated %d piggy banks.', $set->count()));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
private function recalculatePiggyBankEvents(PiggyBank $piggyBank): void
|
|
|
|
{
|
|
|
|
$set = $piggyBank->piggyBankEvents()->get();
|
|
|
|
$set->each(
|
2024-12-21 12:27:07 +01:00
|
|
|
static function (PiggyBankEvent $event): void {
|
2024-12-21 11:20:48 +01:00
|
|
|
$event->touch();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
Log::debug(sprintf('Recalculated %d piggy bank events.', $set->count()));
|
|
|
|
}
|
|
|
|
|
|
|
|
private function recalculateBudgets(UserGroup $userGroup, TransactionCurrency $currency): void
|
|
|
|
{
|
|
|
|
$set = $userGroup->budgets()->get();
|
2024-12-21 12:27:07 +01:00
|
|
|
|
2024-12-21 11:20:48 +01:00
|
|
|
/** @var Budget $budget */
|
|
|
|
foreach ($set as $budget) {
|
|
|
|
$this->recalculateBudgetLimits($budget, $currency);
|
|
|
|
$this->recalculateAutoBudgets($budget, $currency);
|
|
|
|
}
|
|
|
|
Log::debug(sprintf('Recalculated %d budgets.', $set->count()));
|
|
|
|
}
|
|
|
|
|
|
|
|
private function recalculateBudgetLimits(Budget $budget, TransactionCurrency $currency): void
|
|
|
|
{
|
|
|
|
$set = $budget->budgetlimits()->where('transaction_currency_id', '!=', $currency->id)->get();
|
2024-12-21 12:27:07 +01:00
|
|
|
|
2024-12-21 11:20:48 +01:00
|
|
|
/** @var BudgetLimit $limit */
|
|
|
|
foreach ($set as $limit) {
|
|
|
|
Log::debug(sprintf('Will now touch BL #%d', $limit->id));
|
|
|
|
$limit->touch();
|
|
|
|
Log::debug(sprintf('Done with touch BL #%d', $limit->id));
|
|
|
|
}
|
|
|
|
Log::debug(sprintf('Recalculated %d budget limits.', $set->count()));
|
|
|
|
}
|
|
|
|
|
|
|
|
private function recalculateAutoBudgets(Budget $budget, TransactionCurrency $currency): void
|
|
|
|
{
|
|
|
|
$set = $budget->autoBudgets()->where('transaction_currency_id', '!=', $currency->id)->get();
|
2024-12-21 12:27:07 +01:00
|
|
|
|
2024-12-21 11:20:48 +01:00
|
|
|
/** @var AutoBudget $autoBudget */
|
|
|
|
foreach ($set as $autoBudget) {
|
|
|
|
$autoBudget->touch();
|
|
|
|
}
|
|
|
|
Log::debug(sprintf('Recalculated %d auto budgets.', $set->count()));
|
|
|
|
}
|
|
|
|
|
|
|
|
private function recalculateAvailableBudgets(UserGroup $userGroup, TransactionCurrency $currency): void
|
|
|
|
{
|
|
|
|
Log::debug('Start with available budgets.');
|
|
|
|
$set = $userGroup->availableBudgets()->where('transaction_currency_id', '!=', $currency->id)->get();
|
2024-12-21 12:27:07 +01:00
|
|
|
|
2024-12-21 11:20:48 +01:00
|
|
|
/** @var AvailableBudget $budget */
|
|
|
|
foreach ($set as $budget) {
|
|
|
|
$budget->touch();
|
|
|
|
}
|
|
|
|
Log::debug(sprintf('Recalculated %d available budgets.', $set->count()));
|
|
|
|
}
|
|
|
|
|
2024-12-22 08:43:12 +01:00
|
|
|
private function recalculateBills(UserGroup $userGroup, TransactionCurrency $currency): void
|
|
|
|
{
|
|
|
|
$set = $userGroup->bills()->where('transaction_currency_id', '!=', $currency->id)->get();
|
|
|
|
|
|
|
|
/** @var Bill $bill */
|
|
|
|
foreach ($set as $bill) {
|
|
|
|
$bill->touch();
|
|
|
|
}
|
|
|
|
Log::debug(sprintf('Recalculated %d bills.', $set->count()));
|
|
|
|
}
|
|
|
|
|
2024-12-21 11:20:48 +01:00
|
|
|
private function calculateTransactions(UserGroup $userGroup, TransactionCurrency $currency): void
|
|
|
|
{
|
|
|
|
// custom query because of the potential size of this update.
|
2024-12-21 12:27:07 +01:00
|
|
|
$set = DB::table('transactions')
|
|
|
|
->join('transaction_journals', 'transaction_journals.id', '=', 'transactions.transaction_journal_id')
|
|
|
|
->where('transaction_journals.user_group_id', $userGroup->id)
|
|
|
|
->where(static function (DatabaseBuilder $q) use ($currency): void {
|
|
|
|
$q->whereNot('transactions.transaction_currency_id', $currency->id)
|
|
|
|
->orWhereNot('transactions.foreign_currency_id', $currency->id)
|
|
|
|
;
|
|
|
|
})
|
|
|
|
->get(['transactions.id'])
|
|
|
|
;
|
2024-12-21 11:33:58 +01:00
|
|
|
TransactionObserver::$recalculate = false;
|
2024-12-21 11:20:48 +01:00
|
|
|
foreach ($set as $item) {
|
|
|
|
// here we are.
|
|
|
|
$transaction = Transaction::find($item->id);
|
|
|
|
$transaction->touch();
|
|
|
|
}
|
2024-12-21 11:33:58 +01:00
|
|
|
TransactionObserver::$recalculate = true;
|
2024-12-21 11:20:48 +01:00
|
|
|
Log::debug(sprintf('Recalculated %d transactions.', $set->count()));
|
|
|
|
}
|
|
|
|
}
|