mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-25 21:16:47 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			633 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			633 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * RecurringRepository.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\Repositories\Recurring;
 | |
| 
 | |
| use Carbon\Carbon;
 | |
| use FireflyIII\Exceptions\FireflyException;
 | |
| use FireflyIII\Factory\RecurrenceFactory;
 | |
| use FireflyIII\Helpers\Collector\GroupCollectorInterface;
 | |
| use FireflyIII\Models\Note;
 | |
| use FireflyIII\Models\Preference;
 | |
| use FireflyIII\Models\Recurrence;
 | |
| use FireflyIII\Models\RecurrenceMeta;
 | |
| use FireflyIII\Models\RecurrenceRepetition;
 | |
| use FireflyIII\Models\RecurrenceTransaction;
 | |
| use FireflyIII\Models\RecurrenceTransactionMeta;
 | |
| use FireflyIII\Models\TransactionJournal;
 | |
| use FireflyIII\Models\TransactionJournalMeta;
 | |
| use FireflyIII\Services\Internal\Destroy\RecurrenceDestroyService;
 | |
| use FireflyIII\Services\Internal\Update\RecurrenceUpdateService;
 | |
| use FireflyIII\Support\Repositories\Recurring\CalculateRangeOccurrences;
 | |
| use FireflyIII\Support\Repositories\Recurring\CalculateXOccurrences;
 | |
| use FireflyIII\Support\Repositories\Recurring\CalculateXOccurrencesSince;
 | |
| use FireflyIII\Support\Repositories\Recurring\FiltersWeekends;
 | |
| use FireflyIII\User;
 | |
| use Illuminate\Pagination\LengthAwarePaginator;
 | |
| use Illuminate\Support\Collection;
 | |
| use JsonException;
 | |
| use Log;
 | |
| 
 | |
| /**
 | |
|  * Class RecurringRepository
 | |
|  */
 | |
| class RecurringRepository implements RecurringRepositoryInterface
 | |
| {
 | |
|     use CalculateRangeOccurrences, CalculateXOccurrences, CalculateXOccurrencesSince, FiltersWeekends;
 | |
| 
 | |
|     private User $user;
 | |
| 
 | |
|     /**
 | |
|      * Destroy a recurring transaction.
 | |
|      *
 | |
|      * @param Recurrence $recurrence
 | |
|      */
 | |
|     public function destroy(Recurrence $recurrence): void
 | |
|     {
 | |
|         /** @var RecurrenceDestroyService $service */
 | |
|         $service = app(RecurrenceDestroyService::class);
 | |
|         $service->destroy($recurrence);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @inheritDoc
 | |
|      */
 | |
|     public function destroyAll(): void
 | |
|     {
 | |
|         $this->user->recurrences()->delete();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns all of the user's recurring transactions.
 | |
|      *
 | |
|      * @return Collection
 | |
|      */
 | |
|     public function get(): Collection
 | |
|     {
 | |
|         return $this->user->recurrences()
 | |
|                           ->with(['TransactionCurrency', 'TransactionType', 'RecurrenceRepetitions', 'RecurrenceTransactions'])
 | |
|                           ->orderBy('active', 'DESC')
 | |
|                           ->orderBy('transaction_type_id', 'ASC')
 | |
|                           ->orderBy('title', 'ASC')
 | |
|                           ->get();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get ALL recurring transactions.
 | |
|      *
 | |
|      * @return Collection
 | |
|      */
 | |
|     public function getAll(): Collection
 | |
|     {
 | |
|         // grab ALL recurring transactions:
 | |
|         return Recurrence
 | |
|             ::with(['TransactionCurrency', 'TransactionType', 'RecurrenceRepetitions', 'RecurrenceTransactions'])
 | |
|             ->orderBy('active', 'DESC')
 | |
|             ->orderBy('title', 'ASC')
 | |
|             ->get();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @inheritDoc
 | |
|      */
 | |
|     public function getBillId(RecurrenceTransaction $recTransaction): ?int
 | |
|     {
 | |
|         $return = null;
 | |
|         /** @var RecurrenceTransactionMeta $meta */
 | |
|         foreach ($recTransaction->recurrenceTransactionMeta as $meta) {
 | |
|             if ('bill_id' === $meta->name) {
 | |
|                 $return = (int) $meta->value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $return;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the budget ID from a recurring transaction transaction.
 | |
|      *
 | |
|      * @param RecurrenceTransaction $recTransaction
 | |
|      *
 | |
|      * @return null|int
 | |
|      */
 | |
|     public function getBudget(RecurrenceTransaction $recTransaction): ?int
 | |
|     {
 | |
|         $return = 0;
 | |
|         /** @var RecurrenceTransactionMeta $meta */
 | |
|         foreach ($recTransaction->recurrenceTransactionMeta as $meta) {
 | |
|             if ('budget_id' === $meta->name) {
 | |
|                 $return = (int) $meta->value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return 0 === $return ? null : $return;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the category from a recurring transaction transaction.
 | |
|      *
 | |
|      * @param RecurrenceTransaction $recTransaction
 | |
|      *
 | |
|      * @return null|int
 | |
|      */
 | |
|     public function getCategoryId(RecurrenceTransaction $recTransaction): ?int
 | |
|     {
 | |
|         $return = '';
 | |
|         /** @var RecurrenceTransactionMeta $meta */
 | |
|         foreach ($recTransaction->recurrenceTransactionMeta as $meta) {
 | |
|             if ('category_id' === $meta->name) {
 | |
|                 $return = (int) $meta->value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return '' === $return ? null : $return;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the category from a recurring transaction transaction.
 | |
|      *
 | |
|      * @param RecurrenceTransaction $recTransaction
 | |
|      *
 | |
|      * @return null|string
 | |
|      */
 | |
|     public function getCategoryName(RecurrenceTransaction $recTransaction): ?string
 | |
|     {
 | |
|         $return = '';
 | |
|         /** @var RecurrenceTransactionMeta $meta */
 | |
|         foreach ($recTransaction->recurrenceTransactionMeta as $meta) {
 | |
|             if ('category_name' === $meta->name) {
 | |
|                 $return = (string) $meta->value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return '' === $return ? null : $return;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns the journals created for this recurrence, possibly limited by time.
 | |
|      *
 | |
|      * @param Recurrence  $recurrence
 | |
|      * @param Carbon|null $start
 | |
|      * @param Carbon|null $end
 | |
|      *
 | |
|      * @return int
 | |
|      */
 | |
|     public function getJournalCount(Recurrence $recurrence, Carbon $start = null, Carbon $end = null): int
 | |
|     {
 | |
|         $query = TransactionJournal
 | |
|             ::leftJoin('journal_meta', 'journal_meta.transaction_journal_id', '=', 'transaction_journals.id')
 | |
|             ->where('transaction_journals.user_id', $recurrence->user_id)
 | |
|             ->whereNull('transaction_journals.deleted_at')
 | |
|             ->where('journal_meta.name', 'recurrence_id')
 | |
|             ->where('journal_meta.data', '"' . $recurrence->id . '"');
 | |
|         if (null !== $start) {
 | |
|             $query->where('transaction_journals.date', '>=', $start->format('Y-m-d 00:00:00'));
 | |
|         }
 | |
|         if (null !== $end) {
 | |
|             $query->where('transaction_journals.date', '<=', $end->format('Y-m-d 00:00:00'));
 | |
|         }
 | |
| 
 | |
|         return $query->count(['transaction_journals.id']);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get journal ID's for journals created by this recurring transaction.
 | |
|      *
 | |
|      * @param Recurrence $recurrence
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     public function getJournalIds(Recurrence $recurrence): array
 | |
|     {
 | |
|         return TransactionJournalMeta::leftJoin('transaction_journals', 'transaction_journals.id', '=', 'journal_meta.transaction_journal_id')
 | |
|                                      ->where('transaction_journals.user_id', $this->user->id)
 | |
|                                      ->where('journal_meta.name', '=', 'recurrence_id')
 | |
|                                      ->where('journal_meta.data', '=', json_encode((string) $recurrence->id))
 | |
|                                      ->get(['journal_meta.transaction_journal_id'])->pluck('transaction_journal_id')->toArray();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the notes.
 | |
|      *
 | |
|      * @param Recurrence $recurrence
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     public function getNoteText(Recurrence $recurrence): string
 | |
|     {
 | |
|         /** @var Note $note */
 | |
|         $note = $recurrence->notes()->first();
 | |
|         if (null !== $note) {
 | |
|             return (string) $note->text;
 | |
|         }
 | |
| 
 | |
|         return '';
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param RecurrenceTransaction $transaction
 | |
|      *
 | |
|      * @return int|null
 | |
|      */
 | |
|     public function getPiggyBank(RecurrenceTransaction $transaction): ?int
 | |
|     {
 | |
|         $meta = $transaction->recurrenceTransactionMeta;
 | |
|         /** @var RecurrenceTransactionMeta $metaEntry */
 | |
|         foreach ($meta as $metaEntry) {
 | |
|             if ('piggy_bank_id' === $metaEntry->name) {
 | |
|                 return (int) $metaEntry->value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return null;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the tags from the recurring transaction.
 | |
|      *
 | |
|      * @param RecurrenceTransaction $transaction
 | |
|      *
 | |
|      * @return array
 | |
|      * @throws JsonException
 | |
|      */
 | |
|     public function getTags(RecurrenceTransaction $transaction): array
 | |
|     {
 | |
|         $tags = [];
 | |
|         /** @var RecurrenceMeta $meta */
 | |
|         foreach ($transaction->recurrenceTransactionMeta as $meta) {
 | |
|             if ('tags' === $meta->name && '' !== $meta->value) {
 | |
|                 $tags = json_decode($meta->value, true, 512, JSON_THROW_ON_ERROR);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $tags;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Recurrence $recurrence
 | |
|      * @param int        $page
 | |
|      * @param int        $pageSize
 | |
|      *
 | |
|      * @return LengthAwarePaginator
 | |
|      */
 | |
|     public function getTransactionPaginator(Recurrence $recurrence, int $page, int $pageSize): LengthAwarePaginator
 | |
|     {
 | |
|         $journalMeta = TransactionJournalMeta
 | |
|             ::leftJoin('transaction_journals', 'transaction_journals.id', '=', 'journal_meta.transaction_journal_id')
 | |
|             ->whereNull('transaction_journals.deleted_at')
 | |
|             ->where('transaction_journals.user_id', $this->user->id)
 | |
|             ->where('name', 'recurrence_id')
 | |
|             ->where('data', json_encode((string) $recurrence->id))
 | |
|             ->get()->pluck('transaction_journal_id')->toArray();
 | |
|         $search      = [];
 | |
|         foreach ($journalMeta as $journalId) {
 | |
|             $search[] = (int) $journalId;
 | |
|         }
 | |
|         /** @var GroupCollectorInterface $collector */
 | |
|         $collector = app(GroupCollectorInterface::class);
 | |
| 
 | |
|         $collector->setUser($recurrence->user);
 | |
|         $collector->withCategoryInformation()->withBudgetInformation()->setLimit($pageSize)->setPage($page)
 | |
|                   ->withAccountInformation();
 | |
|         $collector->setJournalIds($search);
 | |
| 
 | |
|         return $collector->getPaginatedGroups();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Recurrence $recurrence
 | |
|      *
 | |
|      * @return Collection
 | |
|      */
 | |
|     public function getTransactions(Recurrence $recurrence): Collection
 | |
|     {
 | |
|         $journalMeta = TransactionJournalMeta
 | |
|             ::leftJoin('transaction_journals', 'transaction_journals.id', '=', 'journal_meta.transaction_journal_id')
 | |
|             ->whereNull('transaction_journals.deleted_at')
 | |
|             ->where('transaction_journals.user_id', $this->user->id)
 | |
|             ->where('name', 'recurrence_id')
 | |
|             ->where('data', json_encode((string) $recurrence->id))
 | |
|             ->get()->pluck('transaction_journal_id')->toArray();
 | |
|         $search      = [];
 | |
| 
 | |
|         foreach ($journalMeta as $journalId) {
 | |
|             $search[] = (int) $journalId;
 | |
|         }
 | |
|         if (empty($search)) {
 | |
| 
 | |
|             return new Collection;
 | |
|         }
 | |
| 
 | |
|         /** @var GroupCollectorInterface $collector */
 | |
|         $collector = app(GroupCollectorInterface::class);
 | |
| 
 | |
|         $collector->setUser($recurrence->user);
 | |
|         $collector->withCategoryInformation()->withBudgetInformation()->withAccountInformation();
 | |
|         // filter on specific journals.
 | |
|         $collector->setJournalIds($search);
 | |
| 
 | |
|         return $collector->getGroups();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Calculate the next X iterations starting on the date given in $date.
 | |
|      *
 | |
|      * @param RecurrenceRepetition $repetition
 | |
|      * @param Carbon               $date
 | |
|      * @param int                  $count
 | |
|      *
 | |
|      * @return array
 | |
|      *
 | |
|      */
 | |
|     public function getXOccurrences(RecurrenceRepetition $repetition, Carbon $date, int $count): array
 | |
|     {
 | |
|         $skipMod     = $repetition->repetition_skip + 1;
 | |
|         $occurrences = [];
 | |
|         if ('daily' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXDailyOccurrences($date, $count, $skipMod);
 | |
|         }
 | |
|         if ('weekly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXWeeklyOccurrences($date, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('monthly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXMonthlyOccurrences($date, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('ndom' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXNDomOccurrences($date, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('yearly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXYearlyOccurrences($date, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
| 
 | |
|         // filter out all the weekend days:
 | |
|         return $this->filterWeekends($repetition, $occurrences);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Calculate the next X iterations starting on the date given in $date.
 | |
|      * Returns an array of Carbon objects.
 | |
|      *
 | |
|      * Only returns them of they are after $afterDate
 | |
|      *
 | |
|      * @param RecurrenceRepetition $repetition
 | |
|      * @param Carbon               $date
 | |
|      * @param Carbon               $afterDate
 | |
|      * @param int                  $count
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     public function getXOccurrencesSince(RecurrenceRepetition $repetition, Carbon $date, Carbon $afterDate, int $count): array
 | |
|     {
 | |
|         Log::debug('Now in getXOccurrencesSince()');
 | |
|         $skipMod     = $repetition->repetition_skip + 1;
 | |
|         $occurrences = [];
 | |
|         if ('daily' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXDailyOccurrencesSince($date, $afterDate, $count, $skipMod);
 | |
|         }
 | |
|         if ('weekly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXWeeklyOccurrencesSince($date, $afterDate, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('monthly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXMonthlyOccurrencesSince($date, $afterDate, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('ndom' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXNDomOccurrencesSince($date, $afterDate, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('yearly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getXYearlyOccurrencesSince($date, $afterDate, $count, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
| 
 | |
|         // filter out all the weekend days:
 | |
|         $occurrences = $this->filterWeekends($repetition, $occurrences);
 | |
| 
 | |
|         // filter out everything if "repeat_until" is set.
 | |
|         $repeatUntil = $repetition->recurrence->repeat_until;
 | |
| 
 | |
|         return $this->filterMaxDate($repeatUntil, $occurrences);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Carbon|null $max
 | |
|      * @param array       $occurrences
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     private function filterMaxDate(?Carbon $max, array $occurrences): array
 | |
|     {
 | |
|         if (null === $max) {
 | |
|             return $occurrences;
 | |
|         }
 | |
|         $filtered = [];
 | |
|         foreach ($occurrences as $date) {
 | |
|             if ($date->lte($max)) {
 | |
|                 $filtered[] = $date;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $filtered;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Parse the repetition in a string that is user readable.
 | |
|      *
 | |
|      * @param RecurrenceRepetition $repetition
 | |
|      *
 | |
|      * @return string
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function repetitionDescription(RecurrenceRepetition $repetition): string
 | |
|     {
 | |
|         Log::debug('Now in repetitionDescription()');
 | |
|         /** @var Preference $pref */
 | |
|         $pref     = app('preferences')->getForUser($this->user, 'language', config('firefly.default_language', 'en_US'));
 | |
|         $language = $pref->data;
 | |
|         if ('daily' === $repetition->repetition_type) {
 | |
|             return (string) trans('firefly.recurring_daily', [], $language);
 | |
|         }
 | |
|         if ('weekly' === $repetition->repetition_type) {
 | |
| 
 | |
|             $dayOfWeek = trans(sprintf('config.dow_%s', $repetition->repetition_moment), [], $language);
 | |
|             if ($repetition->repetition_skip > 0) {
 | |
|                 return (string) trans('firefly.recurring_weekly_skip', ['weekday' => $dayOfWeek, 'skip' => $repetition->repetition_skip + 1], $language);
 | |
|             }
 | |
| 
 | |
|             return (string) trans('firefly.recurring_weekly', ['weekday' => $dayOfWeek], $language);
 | |
|         }
 | |
|         if ('monthly' === $repetition->repetition_type) {
 | |
|             if ($repetition->repetition_skip > 0) {
 | |
|                 return (string) trans(
 | |
|                     'firefly.recurring_monthly_skip', ['dayOfMonth' => $repetition->repetition_moment, 'skip' => $repetition->repetition_skip + 1], $language
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             return (string) trans(
 | |
|                 'firefly.recurring_monthly', ['dayOfMonth' => $repetition->repetition_moment, 'skip' => $repetition->repetition_skip - 1], $language
 | |
|             );
 | |
|         }
 | |
|         if ('ndom' === $repetition->repetition_type) {
 | |
|             $parts = explode(',', $repetition->repetition_moment);
 | |
|             // first part is number of week, second is weekday.
 | |
|             $dayOfWeek = trans(sprintf('config.dow_%s', $parts[1]), [], $language);
 | |
| 
 | |
|             return (string) trans('firefly.recurring_ndom', ['weekday' => $dayOfWeek, 'dayOfMonth' => $parts[0]], $language);
 | |
|         }
 | |
|         if ('yearly' === $repetition->repetition_type) {
 | |
|             //
 | |
|             $today       = Carbon::now()->endOfYear();
 | |
|             $repDate     = Carbon::createFromFormat('Y-m-d', $repetition->repetition_moment);
 | |
|             $diffInYears = $today->diffInYears($repDate);
 | |
|             $repDate->addYears($diffInYears); // technically not necessary.
 | |
|             $string = $repDate->isoFormat((string) trans('config.month_and_day_no_year_js'));
 | |
| 
 | |
|             return (string) trans('firefly.recurring_yearly', ['date' => $string], $language);
 | |
|         }
 | |
| 
 | |
|         return '';
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @inheritDoc
 | |
|      */
 | |
|     public function searchRecurrence(string $query, int $limit): Collection
 | |
|     {
 | |
|         $search = $this->user->recurrences();
 | |
|         if ('' !== $query) {
 | |
|             $search->where('recurrences.title', 'LIKE', sprintf('%%%s%%', $query));
 | |
|         }
 | |
|         $search
 | |
|             ->orderBy('recurrences.title', 'ASC');
 | |
| 
 | |
|         return $search->take($limit)->get(['id', 'title', 'description']);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set user for in repository.
 | |
|      *
 | |
|      * @param User $user
 | |
|      */
 | |
|     public function setUser(User $user): void
 | |
|     {
 | |
|         $this->user = $user;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param array $data
 | |
|      *
 | |
|      * @return Recurrence
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function store(array $data): Recurrence
 | |
|     {
 | |
|         /** @var RecurrenceFactory $factory */
 | |
|         $factory = app(RecurrenceFactory::class);
 | |
|         $factory->setUser($this->user);
 | |
|         $result = $factory->create($data);
 | |
|         if (null === $result) {
 | |
|             throw new FireflyException($factory->getErrors()->first());
 | |
|         }
 | |
| 
 | |
|         return $result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @inheritDoc
 | |
|      */
 | |
|     public function totalTransactions(Recurrence $recurrence, RecurrenceRepetition $repetition): int
 | |
|     {
 | |
|         // if repeat = null just return 0.
 | |
|         if (null === $recurrence->repeat_until && 0 === (int) $recurrence->repetitions) {
 | |
|             return 0;
 | |
|         }
 | |
|         // expect X transactions then stop. Return that number
 | |
|         if (null === $recurrence->repeat_until && 0 !== (int) $recurrence->repetitions) {
 | |
|             return (int) $recurrence->repetitions;
 | |
|         }
 | |
| 
 | |
|         // need to calculate, this depends on the repetition:
 | |
|         if (null !== $recurrence->repeat_until && 0 === (int) $recurrence->repetitions) {
 | |
|             $occurrences = $this->getOccurrencesInRange($repetition, $recurrence->first_date ?? today(), $recurrence->repeat_until);
 | |
| 
 | |
|             return count($occurrences);
 | |
|         }
 | |
| 
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Generate events in the date range.
 | |
|      *
 | |
|      * @param RecurrenceRepetition $repetition
 | |
|      * @param Carbon               $start
 | |
|      * @param Carbon               $end
 | |
|      *
 | |
|      * @return array
 | |
|      *
 | |
|      */
 | |
|     public function getOccurrencesInRange(RecurrenceRepetition $repetition, Carbon $start, Carbon $end): array
 | |
|     {
 | |
|         $occurrences = [];
 | |
|         $mutator     = clone $start;
 | |
|         $mutator->startOfDay();
 | |
|         $skipMod = $repetition->repetition_skip + 1;
 | |
|         Log::debug(sprintf('Calculating occurrences for rep type "%s"', $repetition->repetition_type));
 | |
|         Log::debug(sprintf('Mutator is now: %s', $mutator->format('Y-m-d')));
 | |
| 
 | |
|         if ('daily' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getDailyInRange($mutator, $end, $skipMod);
 | |
|         }
 | |
|         if ('weekly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getWeeklyInRange($mutator, $end, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('monthly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getMonthlyInRange($mutator, $end, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('ndom' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getNdomInRange($mutator, $end, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
|         if ('yearly' === $repetition->repetition_type) {
 | |
|             $occurrences = $this->getYearlyInRange($mutator, $end, $skipMod, $repetition->repetition_moment);
 | |
|         }
 | |
| 
 | |
|         // filter out all the weekend days:
 | |
|         return $this->filterWeekends($repetition, $occurrences);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Update a recurring transaction.
 | |
|      *
 | |
|      * @param Recurrence $recurrence
 | |
|      * @param array      $data
 | |
|      *
 | |
|      * @return Recurrence
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function update(Recurrence $recurrence, array $data): Recurrence
 | |
|     {
 | |
|         /** @var RecurrenceUpdateService $service */
 | |
|         $service = app(RecurrenceUpdateService::class);
 | |
| 
 | |
|         return $service->update($recurrence, $data);
 | |
|     }
 | |
| }
 |