mirror of
https://github.com/firefly-iii/firefly-iii.git
synced 2024-11-29 04:04:11 -06:00
518 lines
17 KiB
PHP
518 lines
17 KiB
PHP
<?php
|
|
|
|
/**
|
|
* CreateRecurringTransactions.php
|
|
* Copyright (c) 2019 thegrumpydictator@gmail.com
|
|
*
|
|
* 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\Jobs;
|
|
|
|
use Carbon\Carbon;
|
|
use FireflyIII\Events\RequestedReportOnJournals;
|
|
use FireflyIII\Events\StoredTransactionGroup;
|
|
use FireflyIII\Factory\PiggyBankEventFactory;
|
|
use FireflyIII\Models\Recurrence;
|
|
use FireflyIII\Models\RecurrenceRepetition;
|
|
use FireflyIII\Models\RecurrenceTransaction;
|
|
use FireflyIII\Models\TransactionGroup;
|
|
use FireflyIII\Models\TransactionJournal;
|
|
use FireflyIII\Repositories\Journal\JournalRepositoryInterface;
|
|
use FireflyIII\Repositories\Recurring\RecurringRepositoryInterface;
|
|
use FireflyIII\Repositories\TransactionGroup\TransactionGroupRepositoryInterface;
|
|
use Illuminate\Bus\Queueable;
|
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
|
use Illuminate\Foundation\Bus\Dispatchable;
|
|
use Illuminate\Queue\InteractsWithQueue;
|
|
use Illuminate\Queue\SerializesModels;
|
|
use Illuminate\Support\Collection;
|
|
use Log;
|
|
|
|
/**
|
|
* Class CreateRecurringTransactions.
|
|
*
|
|
*/
|
|
class CreateRecurringTransactions implements ShouldQueue
|
|
{
|
|
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
|
|
|
|
/** @var Carbon The current date */
|
|
private $date;
|
|
/** @var JournalRepositoryInterface Journal repository */
|
|
private $journalRepository;
|
|
/** @var TransactionGroupRepositoryInterface */
|
|
private $groupRepository;
|
|
/** @var RecurringRepositoryInterface Recurring transactions repository. */
|
|
private $repository;
|
|
/** @var bool Force the transaction to be created no matter what. */
|
|
private $force;
|
|
|
|
/** @var int Number of recurrences submitted */
|
|
public $submitted;
|
|
/** @var int Number of recurrences actually fired */
|
|
public $executed;
|
|
/** @var int Transaction groups created */
|
|
public $created;
|
|
|
|
/**
|
|
* @param Carbon $date
|
|
*/
|
|
public function setDate(Carbon $date): void
|
|
{
|
|
$date->startOfDay();
|
|
$this->date = $date;
|
|
}
|
|
|
|
/**
|
|
* Create a new job instance.
|
|
* @codeCoverageIgnore
|
|
*
|
|
* @param Carbon $date
|
|
*/
|
|
public function __construct(?Carbon $date)
|
|
{
|
|
if (null !== $date) {
|
|
$date->startOfDay();
|
|
$this->date = $date;
|
|
}
|
|
$this->repository = app(RecurringRepositoryInterface::class);
|
|
$this->journalRepository = app(JournalRepositoryInterface::class);
|
|
$this->groupRepository = app(TransactionGroupRepositoryInterface::class);
|
|
$this->force = false;
|
|
$this->submitted = 0;
|
|
$this->executed = 0;
|
|
$this->created = 0;
|
|
|
|
Log::debug(sprintf('Created new CreateRecurringTransactions("%s")', $this->date->format('Y-m-d')));
|
|
|
|
}
|
|
|
|
/**
|
|
* @param bool $force
|
|
*/
|
|
public function setForce(bool $force): void
|
|
{
|
|
$this->force = $force;
|
|
}
|
|
|
|
/**
|
|
* Execute the job.
|
|
*/
|
|
public function handle(): void
|
|
{
|
|
Log::debug(sprintf('Now at start of CreateRecurringTransactions() job for %s.', $this->date->format('D d M Y')));
|
|
$recurrences = $this->repository->getAll();
|
|
$result = [];
|
|
$count = $recurrences->count();
|
|
$this->submitted = $count;
|
|
Log::debug(sprintf('Count of collection is %d', $count));
|
|
|
|
// filter recurrences:
|
|
$filtered = $this->filterRecurrences($recurrences);
|
|
Log::debug(sprintf('Left after filtering is %d', $filtered->count()));
|
|
/** @var Recurrence $recurrence */
|
|
foreach ($filtered as $recurrence) {
|
|
if (!isset($result[$recurrence->user_id])) {
|
|
$result[$recurrence->user_id] = new Collection;
|
|
}
|
|
$this->repository->setUser($recurrence->user);
|
|
$this->journalRepository->setUser($recurrence->user);
|
|
$this->groupRepository->setUser($recurrence->user);
|
|
|
|
// clear cache for user
|
|
app('preferences')->setForUser($recurrence->user, 'lastActivity', microtime());
|
|
|
|
Log::debug(sprintf('Now at recurrence #%d', $recurrence->id));
|
|
$created = $this->handleRepetitions($recurrence);
|
|
Log::debug(sprintf('Done with recurrence #%d', $recurrence->id));
|
|
$result[$recurrence->user_id] = $result[$recurrence->user_id]->merge($created);
|
|
$this->executed++;
|
|
}
|
|
|
|
Log::debug('Now running report thing.');
|
|
// will now send email to users.
|
|
foreach ($result as $userId => $journals) {
|
|
event(new RequestedReportOnJournals($userId, $journals));
|
|
}
|
|
|
|
Log::debug('Done with handle()');
|
|
|
|
// clear cache:
|
|
app('preferences')->mark();
|
|
}
|
|
|
|
/**
|
|
* Return recurring transaction is active.
|
|
*
|
|
* @param Recurrence $recurrence
|
|
*
|
|
* @return bool
|
|
*/
|
|
private function active(Recurrence $recurrence): bool
|
|
{
|
|
return $recurrence->active;
|
|
}
|
|
|
|
/**
|
|
* Helper function for debug information.
|
|
*
|
|
* @param array $occurrences
|
|
*
|
|
* @return array
|
|
*/
|
|
private function debugArray(array $occurrences): array
|
|
{
|
|
$return = [];
|
|
foreach ($occurrences as $entry) {
|
|
$return[] = $entry->format('Y-m-d');
|
|
}
|
|
|
|
return $return;
|
|
}
|
|
|
|
/**
|
|
* Get the start date of a recurrence.
|
|
*
|
|
* @param Recurrence $recurrence
|
|
*
|
|
* @return Carbon
|
|
*/
|
|
private function getStartDate(Recurrence $recurrence): Carbon
|
|
{
|
|
$startDate = clone $recurrence->first_date;
|
|
if (null !== $recurrence->latest_date && $recurrence->latest_date->gte($startDate)) {
|
|
$startDate = clone $recurrence->latest_date;
|
|
}
|
|
|
|
return $startDate;
|
|
}
|
|
|
|
/**
|
|
* Get transaction information from a recurring transaction.
|
|
*
|
|
* @param Recurrence $recurrence
|
|
* @param Carbon $date
|
|
*
|
|
* @return array
|
|
*
|
|
*/
|
|
private function getTransactionData(Recurrence $recurrence, Carbon $date): array
|
|
{
|
|
$transactions = $recurrence->recurrenceTransactions()->get();
|
|
$return = [];
|
|
/** @var RecurrenceTransaction $transaction */
|
|
foreach ($transactions as $index => $transaction) {
|
|
$single = [
|
|
'type' => strtolower($recurrence->transactionType->type),
|
|
'date' => $date,
|
|
'user' => $recurrence->user_id,
|
|
'currency_id' => (int)$transaction->transaction_currency_id,
|
|
'currency_code' => null,
|
|
'description' => $recurrence->recurrenceTransactions()->first()->description,
|
|
'amount' => $transaction->amount,
|
|
'budget_id' => $this->repository->getBudget($transaction),
|
|
'budget_name' => null,
|
|
'category_id' => null,
|
|
'category_name' => $this->repository->getCategory($transaction),
|
|
'source_id' => $transaction->source_id,
|
|
'source_name' => null,
|
|
'destination_id' => $transaction->destination_id,
|
|
'destination_name' => null,
|
|
'foreign_currency_id' => $transaction->foreign_currency_id,
|
|
'foreign_currency_code' => null,
|
|
'foreign_amount' => $transaction->foreign_amount,
|
|
'reconciled' => false,
|
|
'identifier' => $index,
|
|
'recurrence_id' => (int)$recurrence->id,
|
|
'order' => $index,
|
|
'notes' => (string)trans('firefly.created_from_recurrence', ['id' => $recurrence->id, 'title' => $recurrence->title]),
|
|
'tags' => $this->repository->getTags($transaction),
|
|
'piggy_bank_id' => $this->repository->getPiggyBank($transaction),
|
|
'piggy_bank_name' => null,
|
|
'bill_id' => null,
|
|
'bill_name' => null,
|
|
];
|
|
$return[] = $single;
|
|
}
|
|
|
|
return $return;
|
|
}
|
|
|
|
/**
|
|
* Check if the occurences should be executed.
|
|
*
|
|
* @param Recurrence $recurrence
|
|
* @param array $occurrences
|
|
*
|
|
* @return Collection
|
|
*/
|
|
private function handleOccurrences(Recurrence $recurrence, array $occurrences): Collection
|
|
{
|
|
$collection = new Collection;
|
|
/** @var Carbon $date */
|
|
foreach ($occurrences as $date) {
|
|
$result = $this->handleOccurrence($recurrence, $date);
|
|
if (null !== $result) {
|
|
$collection->push($result);
|
|
}
|
|
}
|
|
|
|
return $collection;
|
|
}
|
|
|
|
/**
|
|
* @param Recurrence $recurrence
|
|
* @param Carbon $date
|
|
* @return TransactionGroup|null
|
|
*/
|
|
private function handleOccurrence(Recurrence $recurrence, Carbon $date): ?TransactionGroup
|
|
{
|
|
Log::debug(sprintf('Now at date %s.', $date->format('Y-m-d')));
|
|
$date->startOfDay();
|
|
if ($date->ne($this->date)) {
|
|
Log::debug(sprintf('%s is not today (%s)', $date->format('Y-m-d'), $this->date->format('Y-m-d')));
|
|
|
|
return null;
|
|
}
|
|
Log::debug(sprintf('%s IS today (%s)', $date->format('Y-m-d'), $this->date->format('Y-m-d')));
|
|
|
|
// count created journals on THIS day.
|
|
$journalCount = $this->repository->getJournalCount($recurrence, $date, $date);
|
|
if ($journalCount > 0 && false === $this->force) {
|
|
Log::info(sprintf('Already created %d journal(s) for date %s', $journalCount, $date->format('Y-m-d')));
|
|
|
|
return null;
|
|
}
|
|
|
|
if ($journalCount > 0 && true === $this->force) {
|
|
Log::warning(sprintf('Already created %d groups for date %s but FORCED to continue.', $journalCount, $date->format('Y-m-d')));
|
|
}
|
|
|
|
// create transaction array and send to factory.
|
|
$groupTitle = null;
|
|
if ($recurrence->recurrenceTransactions->count() > 1) {
|
|
/** @var RecurrenceTransaction $first */
|
|
// @codeCoverageIgnoreStart
|
|
$first = $recurrence->recurrenceTransactions()->first();
|
|
$groupTitle = $first->description;
|
|
// @codeCoverageIgnoreEnd
|
|
}
|
|
$array = [
|
|
'user' => $recurrence->user_id,
|
|
'group_title' => $groupTitle,
|
|
'transactions' => $this->getTransactionData($recurrence, $date),
|
|
];
|
|
/** @var TransactionGroup $group */
|
|
$group = $this->groupRepository->store($array);
|
|
$this->created++;
|
|
Log::info(sprintf('Created new transaction group #%d', $group->id));
|
|
|
|
// link to piggy:
|
|
//$this->linkGroupToPiggies($recurrence, $group);
|
|
|
|
// trigger event:
|
|
event(new StoredTransactionGroup($group, $recurrence->apply_rules));
|
|
|
|
// update recurring thing:
|
|
$recurrence->latest_date = $date;
|
|
$recurrence->save();
|
|
|
|
return $group;
|
|
}
|
|
|
|
/**
|
|
* Separate method that will loop all repetitions and do something with it. Will return
|
|
* all created transaction journals.
|
|
*
|
|
* @param Recurrence $recurrence
|
|
*
|
|
* @return Collection
|
|
*/
|
|
private function handleRepetitions(Recurrence $recurrence): Collection
|
|
{
|
|
$collection = new Collection;
|
|
/** @var RecurrenceRepetition $repetition */
|
|
foreach ($recurrence->recurrenceRepetitions as $repetition) {
|
|
Log::debug(
|
|
sprintf(
|
|
'Now repeating %s with value "%s", skips every %d time(s)', $repetition->repetition_type, $repetition->repetition_moment,
|
|
$repetition->repetition_skip
|
|
)
|
|
);
|
|
|
|
// start looping from $startDate to today perhaps we have a hit?
|
|
// add two days to $this->date so we always include the weekend.
|
|
$includeWeekend = clone $this->date;
|
|
$includeWeekend->addDays(2);
|
|
$occurrences = $this->repository->getOccurrencesInRange($repetition, $recurrence->first_date, $includeWeekend);
|
|
Log::debug(
|
|
sprintf(
|
|
'Calculated %d occurrences between %s and %s',
|
|
count($occurrences),
|
|
$recurrence->first_date->format('Y-m-d'),
|
|
$includeWeekend->format('Y-m-d')
|
|
), $this->debugArray($occurrences)
|
|
);
|
|
unset($includeWeekend);
|
|
|
|
$result = $this->handleOccurrences($recurrence, $occurrences);
|
|
$collection = $collection->merge($result);
|
|
}
|
|
|
|
return $collection;
|
|
}
|
|
|
|
/**
|
|
* Has the recurrence fired today.
|
|
*
|
|
* @param Recurrence $recurrence
|
|
*
|
|
* @return bool
|
|
*/
|
|
private function hasFiredToday(Recurrence $recurrence): bool
|
|
{
|
|
return null !== $recurrence->latest_date && $recurrence->latest_date->eq($this->date);
|
|
}
|
|
|
|
/**
|
|
* Has the reuccrence started yet.
|
|
*
|
|
* @param $recurrence
|
|
*
|
|
* @return bool
|
|
*/
|
|
private function hasNotStartedYet(Recurrence $recurrence): bool
|
|
{
|
|
$startDate = $this->getStartDate($recurrence);
|
|
|
|
return $startDate->gt($this->date);
|
|
}
|
|
|
|
/**
|
|
* Return true if the $repeat_until date is in the past.
|
|
*
|
|
* @param Recurrence $recurrence
|
|
*
|
|
* @return bool
|
|
*/
|
|
private function repeatUntilHasPassed(Recurrence $recurrence): bool
|
|
{
|
|
// date has passed
|
|
return null !== $recurrence->repeat_until && $recurrence->repeat_until->lt($this->date);
|
|
}
|
|
|
|
/**
|
|
* Is the info in the recurrence valid?
|
|
*
|
|
* @param Recurrence $recurrence
|
|
*
|
|
* @return bool
|
|
*
|
|
*/
|
|
private function validRecurrence(Recurrence $recurrence): bool
|
|
{
|
|
// is not active.
|
|
if (!$this->active($recurrence)) {
|
|
Log::info(sprintf('Recurrence #%d is not active. Skipped.', $recurrence->id));
|
|
|
|
return false;
|
|
}
|
|
|
|
// has repeated X times.
|
|
$journalCount = $this->repository->getJournalCount($recurrence);
|
|
if (0 !== $recurrence->repetitions && $journalCount >= $recurrence->repetitions && false === $this->force) {
|
|
Log::info(sprintf('Recurrence #%d has run %d times, so will run no longer.', $recurrence->id, $recurrence->repetitions));
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
// is no longer running
|
|
if ($this->repeatUntilHasPassed($recurrence)) {
|
|
Log::info(
|
|
sprintf(
|
|
'Recurrence #%d was set to run until %s, and today\'s date is %s. Skipped.',
|
|
$recurrence->id,
|
|
$recurrence->repeat_until->format('Y-m-d'),
|
|
$this->date->format('Y-m-d')
|
|
)
|
|
);
|
|
|
|
return false;
|
|
}
|
|
|
|
// first_date is in the future
|
|
if ($this->hasNotStartedYet($recurrence)) {
|
|
Log::info(
|
|
sprintf(
|
|
'Recurrence #%d is set to run on %s, and today\'s date is %s. Skipped.',
|
|
$recurrence->id,
|
|
$recurrence->first_date->format('Y-m-d'),
|
|
$this->date->format('Y-m-d')
|
|
)
|
|
);
|
|
|
|
return false;
|
|
}
|
|
|
|
// already fired today (with success):
|
|
if (false === $this->force && $this->hasFiredToday($recurrence)) {
|
|
Log::info(sprintf('Recurrence #%d has already fired today. Skipped.', $recurrence->id));
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
/**
|
|
* @param Collection $recurrences
|
|
* @return Collection
|
|
*/
|
|
private function filterRecurrences(Collection $recurrences): Collection
|
|
{
|
|
return $recurrences->filter(
|
|
function (Recurrence $recurrence) {
|
|
return $this->validRecurrence($recurrence);
|
|
}
|
|
);
|
|
}
|
|
|
|
/***
|
|
* @param Recurrence $recurrence
|
|
* @param TransactionGroup $group
|
|
*/
|
|
private function linkGroupToPiggies(Recurrence $recurrence, TransactionGroup $group): void
|
|
{
|
|
/** @var TransactionJournal $journal */
|
|
foreach ($group->transactionJournals as $journal) {
|
|
// get piggy bank ID from meta data:
|
|
$piggyBank = $this->repository->getPiggyBank($recurrence);
|
|
if (null !== $piggyBank) {
|
|
/** @var PiggyBankEventFactory $factory */
|
|
$factory = app(PiggyBankEventFactory::class);
|
|
$factory->create($journal, $piggyBank);
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
}
|