diff --git a/typo3/sysext/scheduler/Classes/Command/SchedulerCommand.php b/typo3/sysext/scheduler/Classes/Command/SchedulerCommand.php index 7c5cee004a76b83731f6d2ee053fd3e35449b180..13264842d22b7a8e9288a7d4ea75dd0ca70b41e4 100644 --- a/typo3/sysext/scheduler/Classes/Command/SchedulerCommand.php +++ b/typo3/sysext/scheduler/Classes/Command/SchedulerCommand.php @@ -32,11 +32,6 @@ use TYPO3\CMS\Scheduler\Validation\Validator\TaskValidator; */ class SchedulerCommand extends Command { - /** - * @var bool - */ - protected $hasTask = true; - /** * @var SymfonyStyle */ @@ -118,8 +113,7 @@ Call it like this: typo3/sysext/core/bin/typo3 scheduler:run --task=13 -f') $this->forceExecution = (bool)$input->getOption('force'); $this->stopTasks = $this->shouldStopTasks((bool)$input->getOption('stop')); - $this->loopTasks(); - return Command::SUCCESS; + return $this->loopTasks() ? Command::SUCCESS : Command::FAILURE; } /** @@ -172,38 +166,37 @@ Call it like this: typo3/sysext/core/bin/typo3 scheduler:run --task=13 -f') /** * Execute tasks in loop that are ready to execute */ - protected function loopTasks() + protected function loopTasks(): bool { + $hasError = false; do { + $task = null; // Try getting the next task and execute it // If there are no more tasks to execute, an exception is thrown by \TYPO3\CMS\Scheduler\Scheduler::fetchTask() try { $task = $this->fetchNextTask(); + if ($task === null) { + break; + } try { $this->executeOrStopTask($task); } catch (\Exception $e) { - if ($this->io->isVerbose()) { - $this->io->warning($e->getMessage()); - } + $this->io->getErrorStyle()->error($e->getMessage()); + $hasError = true; // We ignore any exception that may have been thrown during execution, // as this is a background process. // The exception message has been recorded to the database anyway continue; } - } catch (\OutOfBoundsException $e) { - if ($this->io->isVeryVerbose()) { - $this->io->writeln($e->getMessage()); - } - $this->hasTask = !empty($this->overwrittenTaskList); } catch (\UnexpectedValueException $e) { - if ($this->io->isVerbose()) { - $this->io->warning($e->getMessage()); - } + $this->io->getErrorStyle()->error($e->getMessage()); + $hasError = true; continue; } - } while ($this->hasTask); + } while ($task !== null); // Record the run in the system registry $this->scheduler->recordLastRun(); + return !$hasError; } /** @@ -212,17 +205,16 @@ Call it like this: typo3/sysext/core/bin/typo3 scheduler:run --task=13 -f') * * Without the --task option we ask the scheduler for the next task with pending execution. * - * @throws \OutOfBoundsException When there are no more tasks to execute. * @throws \UnexpectedValueException When no task is found by the provided UID or the task is not marked for execution. */ - protected function fetchNextTask(): AbstractTask + protected function fetchNextTask(): ?AbstractTask { if ($this->overwrittenTaskList === null) { return $this->taskRepository->findNextExecutableTask(); } if (count($this->overwrittenTaskList) === 0) { - throw new \OutOfBoundsException('No more tasks to execute', 1547675594); + return null; } $taskUid = (int)array_shift($this->overwrittenTaskList); diff --git a/typo3/sysext/scheduler/Classes/Domain/Repository/SchedulerTaskRepository.php b/typo3/sysext/scheduler/Classes/Domain/Repository/SchedulerTaskRepository.php index 359361990337e501ae36c7254f2af095bf34a3c6..1d7db9b46d1d0727c175c8c9aad375b1413003f7 100644 --- a/typo3/sysext/scheduler/Classes/Domain/Repository/SchedulerTaskRepository.php +++ b/typo3/sysext/scheduler/Classes/Domain/Repository/SchedulerTaskRepository.php @@ -201,10 +201,9 @@ class SchedulerTaskRepository * next due task is returned. If there are no due tasks the method throws an exception. * * @return AbstractTask The fetched task object - * @throws \OutOfBoundsException * @throws \UnexpectedValueException */ - public function findNextExecutableTask(): AbstractTask + public function findNextExecutableTask(): ?AbstractTask { // If no uid is given, take any non-disabled task which has a next execution time in the past $connectionPool = GeneralUtility::makeInstance(ConnectionPool::class); @@ -241,8 +240,7 @@ class SchedulerTaskRepository $row = $queryBuilder->executeQuery()->fetchAssociative(); if (empty($row)) { - // No uid was passed and no overdue task was found - throw new \OutOfBoundsException('No (more) tasks available for execution', 1247827244); + return null; } return $this->createValidTaskObjectOrDisableTask($row); diff --git a/typo3/sysext/scheduler/Classes/Scheduler.php b/typo3/sysext/scheduler/Classes/Scheduler.php index 84656a6a24959ed67382e28b06964d2318f52809..e756c46714e61dcf091c8820c4cbec288b12bf6a 100644 --- a/typo3/sysext/scheduler/Classes/Scheduler.php +++ b/typo3/sysext/scheduler/Classes/Scheduler.php @@ -256,12 +256,12 @@ class Scheduler implements SingletonInterface * If there are no due tasks the method throws an exception. * * @param int $uid Primary key of a task - * @return Task\AbstractTask The fetched task object + * @return Task\AbstractTask|null The fetched task object * @throws \OutOfBoundsException * @throws \UnexpectedValueException * @deprecated will be removed in TYPO3 v13.0. Use SchedulerTaskRepository instead. */ - public function fetchTask($uid = 0): AbstractTask + public function fetchTask($uid = 0): ?AbstractTask { trigger_error('Scheduler->' . __METHOD__ . ' will be removed in TYPO3 v13.0. Use SchedulerTaskRepository instead.', E_USER_DEPRECATED); if ($uid > 0) {