From 65428ad3f4c9f2c6dd5a14d6882351e55ca4e016 Mon Sep 17 00:00:00 2001 From: Markus Klein <markus.klein@typo3.org> Date: Thu, 23 Feb 2023 13:07:10 +0100 Subject: [PATCH] [BUGFIX] Output errors to CLI in scheduler:run command MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Whenever a scheduler task fails during execution via CLI `scheduler:run`, the error message is now always written to stderr. This gives the task executor (e.g. cron) a chance to identify any failures, if even stdout is redirected to /dev/null. The task-loop is adapted to avoid (ab)using exceptions for control-flow (for loop-abortion). It instead now uses nullables to signal that not finding a next-task is expected behavior, not an exception. Also the exit code of the command is adapted to switch to 1 in case at least one error occured. (This is important in case the cronjob is executed as systemd-timer) Resolves: #87806 Releases: main, 11.5 Change-Id: I0b55bab7fcd916209114d9e3e07f54c7243ce9a5 Reviewed-on: https://review.typo3.org/c/Packages/TYPO3.CMS/+/77936 Reviewed-by: Stephan Großberndt <stephan.grossberndt@typo3.org> Reviewed-by: Markus Klein <markus.klein@typo3.org> Tested-by: Benjamin Franzke <benjaminfranzke@gmail.com> Tested-by: Christian Kuhn <lolli@schwarzbu.ch> Tested-by: core-ci <typo3@b13.com> Reviewed-by: Christoph Lehmann <christoph.lehmann@networkteam.com> Reviewed-by: Christian Kuhn <lolli@schwarzbu.ch> Reviewed-by: Benjamin Franzke <benjaminfranzke@gmail.com> --- .../Classes/Command/SchedulerCommand.php | 38 ++++++++----------- .../Repository/SchedulerTaskRepository.php | 6 +-- typo3/sysext/scheduler/Classes/Scheduler.php | 4 +- 3 files changed, 19 insertions(+), 29 deletions(-) diff --git a/typo3/sysext/scheduler/Classes/Command/SchedulerCommand.php b/typo3/sysext/scheduler/Classes/Command/SchedulerCommand.php index 7c5cee004a76..13264842d22b 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 359361990337..1d7db9b46d1d 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 84656a6a2495..e756c46714e6 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) { -- GitLab