2018-02-22 20:07:14 +01:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* BillUpdateService.php
|
2020-02-16 13:56:35 +01:00
|
|
|
* Copyright (c) 2019 james@firefly-iii.org
|
2018-02-22 20:07:14 +01:00
|
|
|
*
|
2019-10-02 06:37:26 +02:00
|
|
|
* This file is part of Firefly III (https://github.com/firefly-iii).
|
2018-02-22 20:07:14 +01:00
|
|
|
*
|
2019-10-02 06:37:26 +02:00
|
|
|
* 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.
|
2018-02-22 20:07:14 +01:00
|
|
|
*
|
2019-10-02 06:37:26 +02:00
|
|
|
* This program is distributed in the hope that it will be useful,
|
2018-02-22 20:07:14 +01:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2019-10-02 06:37:26 +02:00
|
|
|
* GNU Affero General Public License for more details.
|
2018-02-22 20:07:14 +01:00
|
|
|
*
|
2019-10-02 06:37:26 +02:00
|
|
|
* 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/>.
|
2018-02-22 20:07:14 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace FireflyIII\Services\Internal\Update;
|
|
|
|
|
2018-12-21 15:42:40 +01:00
|
|
|
use FireflyIII\Factory\TransactionCurrencyFactory;
|
2018-02-22 20:07:14 +01:00
|
|
|
use FireflyIII\Models\Bill;
|
2020-03-20 08:41:20 +01:00
|
|
|
use FireflyIII\Models\Rule;
|
|
|
|
use FireflyIII\Models\RuleTrigger;
|
2018-12-21 15:42:40 +01:00
|
|
|
use FireflyIII\Models\TransactionCurrency;
|
2020-03-20 08:41:20 +01:00
|
|
|
use FireflyIII\Repositories\Bill\BillRepositoryInterface;
|
2020-06-30 19:06:05 +02:00
|
|
|
use FireflyIII\Repositories\ObjectGroup\CreatesObjectGroups;
|
2018-02-22 20:07:14 +01:00
|
|
|
use FireflyIII\Services\Internal\Support\BillServiceTrait;
|
2020-03-20 08:41:20 +01:00
|
|
|
use Illuminate\Support\Collection;
|
2018-09-06 12:29:32 +02:00
|
|
|
use Log;
|
2019-02-12 21:49:28 +01:00
|
|
|
|
2018-02-22 20:07:14 +01:00
|
|
|
/**
|
2018-03-02 16:31:02 +01:00
|
|
|
* @codeCoverageIgnore
|
2018-02-22 20:07:14 +01:00
|
|
|
* Class BillUpdateService
|
|
|
|
*/
|
|
|
|
class BillUpdateService
|
|
|
|
{
|
2020-06-30 19:06:05 +02:00
|
|
|
use BillServiceTrait, CreatesObjectGroups;
|
|
|
|
|
|
|
|
protected $user;
|
2018-02-22 20:07:14 +01:00
|
|
|
|
2018-09-06 12:29:32 +02:00
|
|
|
/**
|
|
|
|
* Constructor.
|
|
|
|
*/
|
|
|
|
public function __construct()
|
|
|
|
{
|
2018-12-15 07:59:02 +01:00
|
|
|
if ('testing' === config('app.env')) {
|
2019-06-07 18:20:15 +02:00
|
|
|
Log::warning(sprintf('%s should not be instantiated in the TEST environment!', get_class($this)));
|
2018-09-06 12:29:32 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-22 20:07:14 +01:00
|
|
|
/**
|
2019-06-13 06:39:05 +02:00
|
|
|
* @param Bill $bill
|
2018-02-22 20:07:14 +01:00
|
|
|
* @param array $data
|
2018-03-02 16:31:02 +01:00
|
|
|
*
|
|
|
|
* @return Bill
|
2018-02-22 20:07:14 +01:00
|
|
|
*/
|
|
|
|
public function update(Bill $bill, array $data): Bill
|
|
|
|
{
|
2020-06-30 19:06:05 +02:00
|
|
|
$this->user = $bill->user;
|
2018-12-21 15:42:40 +01:00
|
|
|
/** @var TransactionCurrencyFactory $factory */
|
|
|
|
$factory = app(TransactionCurrencyFactory::class);
|
|
|
|
/** @var TransactionCurrency $currency */
|
2018-12-21 16:38:10 +01:00
|
|
|
$currency = $factory->find($data['currency_id'] ?? null, $data['currency_code'] ?? null);
|
2018-12-21 15:42:40 +01:00
|
|
|
|
2019-02-12 21:49:28 +01:00
|
|
|
if (null === $currency) {
|
2018-12-21 15:42:40 +01:00
|
|
|
// use default currency:
|
|
|
|
$currency = app('amount')->getDefaultCurrencyByUser($bill->user);
|
|
|
|
}
|
|
|
|
|
|
|
|
// enable the currency if it isn't.
|
|
|
|
$currency->enabled = true;
|
|
|
|
$currency->save();
|
|
|
|
|
2020-03-20 08:41:20 +01:00
|
|
|
// old values
|
|
|
|
$oldData = [
|
|
|
|
'name' => $bill->name,
|
|
|
|
'amount_min' => $bill->amount_min,
|
|
|
|
'amount_max' => $bill->amount_max,
|
|
|
|
'transaction_currency_name' => $bill->transactionCurrency->name,
|
|
|
|
];
|
|
|
|
// new values
|
|
|
|
$data['transaction_currency_name'] = $currency->name;
|
|
|
|
$bill->name = $data['name'];
|
|
|
|
$bill->match = $data['match'] ?? $bill->match;
|
|
|
|
$bill->amount_min = $data['amount_min'];
|
|
|
|
$bill->amount_max = $data['amount_max'];
|
|
|
|
$bill->date = $data['date'];
|
|
|
|
$bill->transaction_currency_id = $currency->id;
|
|
|
|
$bill->repeat_freq = $data['repeat_freq'];
|
|
|
|
$bill->skip = $data['skip'];
|
|
|
|
$bill->automatch = true;
|
|
|
|
$bill->active = $data['active'] ?? true;
|
2018-02-22 20:07:14 +01:00
|
|
|
$bill->save();
|
|
|
|
|
|
|
|
// update note:
|
2018-08-18 20:13:26 +02:00
|
|
|
if (isset($data['notes'])) {
|
2020-03-20 08:41:20 +01:00
|
|
|
$this->updateNote($bill, (string) $data['notes']);
|
2018-02-22 20:07:14 +01:00
|
|
|
}
|
|
|
|
|
2018-04-14 20:31:31 +02:00
|
|
|
// update rule actions.
|
2020-03-20 08:41:20 +01:00
|
|
|
$this->updateBillActions($bill, $oldData['name'], $data['name']);
|
|
|
|
$this->updateBillTriggers($bill, $oldData, $data);
|
2018-04-14 20:31:31 +02:00
|
|
|
|
2020-06-30 19:06:05 +02:00
|
|
|
// update using name:
|
|
|
|
$objectGroupTitle = $data['object_group'] ?? '';
|
|
|
|
if ('' !== $objectGroupTitle) {
|
|
|
|
$objectGroup = $this->findOrCreateObjectGroup($objectGroupTitle);
|
|
|
|
if (null !== $objectGroup) {
|
|
|
|
$bill->objectGroups()->sync([$objectGroup->id]);
|
|
|
|
$bill->save();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// try also with ID:
|
|
|
|
$objectGroupId = (int) ($data['object_group_id'] ?? 0);
|
|
|
|
if (0 !== $objectGroupId) {
|
|
|
|
$objectGroup = $this->findObjectGroupById($objectGroupId);
|
|
|
|
if (null !== $objectGroup) {
|
|
|
|
$bill->objectGroups()->sync([$objectGroup->id]);
|
|
|
|
$bill->save();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-22 20:07:14 +01:00
|
|
|
return $bill;
|
|
|
|
}
|
|
|
|
|
2020-03-20 08:41:20 +01:00
|
|
|
/**
|
|
|
|
* @param Bill $bill
|
|
|
|
* @param array $oldData
|
|
|
|
* @param array $newData
|
|
|
|
*/
|
|
|
|
private function updateBillTriggers(Bill $bill, array $oldData, array $newData): void
|
|
|
|
{
|
|
|
|
Log::debug(sprintf('Now in updateBillTriggers(%d, "%s")', $bill->id, $bill->name));
|
|
|
|
/** @var BillRepositoryInterface $repository */
|
|
|
|
$repository = app(BillRepositoryInterface::class);
|
|
|
|
$repository->setUser($bill->user);
|
|
|
|
$rules = $repository->getRulesForBill($bill);
|
|
|
|
if (0 === $rules->count()) {
|
|
|
|
Log::debug('Found no rules.');
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Log::debug(sprintf('Found %d rules', $rules->count()));
|
|
|
|
$fields = [
|
|
|
|
'name' => 'description_contains',
|
|
|
|
'amount_min' => 'amount_more',
|
|
|
|
'amount_max' => 'amount_less',
|
|
|
|
'transaction_currency_name' => 'currency_is'];
|
|
|
|
foreach ($fields as $field => $ruleTriggerKey) {
|
|
|
|
if ($oldData[$field] === $newData[$field]) {
|
|
|
|
Log::debug(sprintf('Field %s is unchanged ("%s"), continue.', $field, $oldData[$field]));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$this->updateRules($rules, $ruleTriggerKey, $oldData[$field], $newData[$field]);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Collection $rules
|
|
|
|
* @param string $key
|
|
|
|
* @param string $oldValue
|
|
|
|
* @param string $newValue
|
|
|
|
*/
|
|
|
|
private function updateRules(Collection $rules, string $key, string $oldValue, string $newValue): void
|
|
|
|
{
|
|
|
|
/** @var Rule $rule */
|
|
|
|
foreach ($rules as $rule) {
|
|
|
|
$trigger = $this->getRuleTrigger($rule, $key);
|
|
|
|
if (null !== $trigger && $trigger->trigger_value === $oldValue) {
|
|
|
|
Log::debug(sprintf('Updated rule trigger #%d from value "%s" to value "%s"', $trigger->id, $oldValue, $newValue));
|
|
|
|
$trigger->trigger_value = $newValue;
|
|
|
|
$trigger->save();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (null !== $trigger && $trigger->trigger_value !== $oldValue && in_array($key, ['amount_more', 'amount_less'], true)
|
|
|
|
&& 0 === bccomp($trigger->trigger_value, $oldValue)) {
|
|
|
|
Log::debug(sprintf('Updated rule trigger #%d from value "%s" to value "%s"', $trigger->id, $oldValue, $newValue));
|
|
|
|
$trigger->trigger_value = $newValue;
|
|
|
|
$trigger->save();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Rule $rule
|
|
|
|
* @param string $key
|
|
|
|
*
|
|
|
|
* @return RuleTrigger|null
|
|
|
|
*/
|
|
|
|
private function getRuleTrigger(Rule $rule, string $key): ?RuleTrigger
|
|
|
|
{
|
|
|
|
return $rule->ruleTriggers()->where('trigger_type', $key)->first();
|
|
|
|
}
|
2018-03-05 19:35:58 +01:00
|
|
|
}
|