1
0
قرینه از https://github.com/matomo-org/matomo.git synced 2025-08-22 06:57:53 +00:00
Files
matomo/plugins/CoreAdminHome/Commands/RunScheduledTasks.php
Stefan Giehl eff138b4c3 Implements system signal handling for archiving interruption / termination (#22487)
* Allow signal handling in console commands

* Stop archiving after receiving SIGINT/SIGTERM

* Add tests for basic core:archive signal handling

* Abort symfony archiving requests for SIGTERM

* Add tests for non-symfony core:archive climulti methods

* Add tests for core:archive sigint fallback if sigterm is not fully supported

* Add tests for signal handling during core:archive init

* Add tests for signal handling during core:archive scheduled tasks

* Stop scheduled tasks after receiving SIGINT/SIGTERM

* Add tests for signal handling during scheduled-tasks:run

* Wrap signal handling support behind feature flag

* Update CHANGELOG.md

---------

Co-authored-by: Marc Neudert <marc@innocraft.com>
2025-02-26 16:08:20 +01:00

113 خطوط
3.0 KiB
PHP

<?php
/**
* Matomo - free/libre analytics platform
*
* @link https://matomo.org
* @license https://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later
*/
namespace Piwik\Plugins\CoreAdminHome\Commands;
use Piwik\Container\StaticContainer;
use Piwik\FrontController;
use Piwik\Log\LoggerInterface;
use Piwik\Plugin\ConsoleCommand;
use Piwik\Scheduler\Scheduler;
class RunScheduledTasks extends ConsoleCommand
{
/**
* @var LoggerInterface
*/
private $logger;
/**
* @var Scheduler|null
*/
private $scheduler = null;
public function __construct(LoggerInterface $logger = null)
{
parent::__construct();
$this->logger = $logger ?: StaticContainer::get(LoggerInterface::class);
}
public function getSystemSignalsToHandle(): array
{
return [\SIGINT, \SIGTERM];
}
public function handleSystemSignal(int $signal): void
{
if (null === $this->scheduler) {
// scheduled tasks have not yet started, stop immediately
exit;
}
$this->logger->info('Received system signal to stop scheduled tasks: ' . $signal);
$this->scheduler->handleSignal($signal);
}
protected function configure()
{
$this->setName('scheduled-tasks:run');
$this->setAliases(array('core:run-scheduled-tasks'));
$this->setDescription('Will run all scheduled tasks due to run at this time.');
$this->addOptionalArgument('task', 'Optionally pass the name of a task to run (will run even if not scheduled to run now)');
$this->addNoValueOption('force', null, 'If set, it will execute all tasks even the ones not due to run at this time.');
}
/**
* Execute command like: ./console core:run-scheduled-tasks
*/
protected function doExecute(): int
{
$input = $this->getInput();
$this->forceRunAllTasksIfRequested();
FrontController::getInstance()->init();
// TODO use dependency injection
$this->scheduler = StaticContainer::get(Scheduler::class);
$task = $input->getArgument('task');
if ($task) {
$this->runSingleTask($this->scheduler, $task);
} else {
$this->scheduler->run();
}
$this->writeSuccessMessage('Scheduled Tasks executed');
return self::SUCCESS;
}
private function forceRunAllTasksIfRequested()
{
$force = $this->getInput()->getOption('force');
if ($force && !defined('DEBUG_FORCE_SCHEDULED_TASKS')) {
define('DEBUG_FORCE_SCHEDULED_TASKS', true);
}
}
private function runSingleTask(Scheduler $scheduler, $task)
{
try {
$message = $scheduler->runTaskNow($task);
} catch (\InvalidArgumentException $e) {
$message = $e->getMessage() . PHP_EOL
. 'Available tasks:' . PHP_EOL
. implode(PHP_EOL, $scheduler->getTaskList());
throw new \Exception($message);
}
$this->getOutput()->writeln($message);
}
}