Initial Upload
This commit is contained in:
309
lib/Entity/Task.php
Normal file
309
lib/Entity/Task.php
Normal file
@@ -0,0 +1,309 @@
|
||||
<?php
|
||||
/*
|
||||
* Copyright (C) 2025 Xibo Signage Ltd
|
||||
*
|
||||
* Xibo - Digital Signage - https://xibosignage.com
|
||||
*
|
||||
* This file is part of Xibo.
|
||||
*
|
||||
* Xibo 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
|
||||
* any later version.
|
||||
*
|
||||
* Xibo 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 Xibo. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
|
||||
namespace Xibo\Entity;
|
||||
use Carbon\Carbon;
|
||||
use Cron\CronExpression;
|
||||
use Xibo\Service\LogServiceInterface;
|
||||
use Xibo\Storage\StorageServiceInterface;
|
||||
use Xibo\Support\Exception\InvalidArgumentException;
|
||||
use Xibo\Support\Exception\NotFoundException;
|
||||
|
||||
/**
|
||||
* Class Task
|
||||
* @package Xibo\XTR
|
||||
*/
|
||||
class Task implements \JsonSerializable
|
||||
{
|
||||
use EntityTrait;
|
||||
|
||||
public static $STATUS_RUNNING = 1;
|
||||
public static $STATUS_IDLE = 2;
|
||||
public static $STATUS_ERROR = 3;
|
||||
public static $STATUS_SUCCESS = 4;
|
||||
public static $STATUS_TIMEOUT = 5;
|
||||
|
||||
public $taskId;
|
||||
public $name;
|
||||
public $configFile;
|
||||
public $class;
|
||||
public $status;
|
||||
public $pid = 0;
|
||||
public $options = [];
|
||||
public $schedule;
|
||||
public $lastRunDt = 0;
|
||||
public $lastRunStartDt;
|
||||
public $lastRunMessage;
|
||||
public $lastRunStatus;
|
||||
public $lastRunDuration = 0;
|
||||
public $lastRunExitCode = 0;
|
||||
public $isActive;
|
||||
public $runNow;
|
||||
|
||||
/**
|
||||
* Command constructor.
|
||||
* @param StorageServiceInterface $store
|
||||
* @param LogServiceInterface $log
|
||||
* @param \Symfony\Component\EventDispatcher\EventDispatcherInterface $dispatcher
|
||||
*/
|
||||
public function __construct($store, $log, $dispatcher)
|
||||
{
|
||||
$this->setCommonDependencies($store, $log, $dispatcher);
|
||||
}
|
||||
|
||||
/**
|
||||
* @return \DateTime|string
|
||||
* @throws \Exception
|
||||
*/
|
||||
public function nextRunDate(): \DateTime|string
|
||||
{
|
||||
try {
|
||||
try {
|
||||
$cron = new CronExpression($this->schedule);
|
||||
} catch (\Exception $e) {
|
||||
// Try and take the first X characters instead.
|
||||
try {
|
||||
$cron = new CronExpression(substr($this->schedule, 0, strlen($this->schedule) - 2));
|
||||
} catch (\Exception) {
|
||||
$this->getLog()->error('nextRunDate: cannot fix CRON syntax error ' . $this->taskId);
|
||||
throw $e;
|
||||
}
|
||||
}
|
||||
|
||||
if ($this->lastRunDt == 0) {
|
||||
return (new \DateTime())->format('U');
|
||||
}
|
||||
|
||||
return $cron->getNextRunDate(\DateTime::createFromFormat('U', $this->lastRunDt))->format('U');
|
||||
} catch (\Exception) {
|
||||
$this->getLog()->error('Invalid CRON expression for TaskId ' . $this->taskId);
|
||||
|
||||
$this->status = self::$STATUS_ERROR;
|
||||
return (new \DateTime())->add(new \DateInterval('P1Y'))->format('U');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Set class and options
|
||||
* @throws NotFoundException
|
||||
*/
|
||||
public function setClassAndOptions()
|
||||
{
|
||||
if ($this->configFile == null)
|
||||
throw new NotFoundException(__('No config file recorded for task. Please recreate.'));
|
||||
|
||||
// Get the class and default set of options from the config file.
|
||||
if (!file_exists(PROJECT_ROOT . $this->configFile))
|
||||
throw new NotFoundException(__('Config file not found for Task'));
|
||||
|
||||
$config = json_decode(file_get_contents(PROJECT_ROOT . $this->configFile), true);
|
||||
$this->class = $config['class'];
|
||||
$this->options = array_merge($config['options'], $this->options);
|
||||
}
|
||||
|
||||
/**
|
||||
* Validate
|
||||
* @throws InvalidArgumentException
|
||||
*/
|
||||
private function validate(): void
|
||||
{
|
||||
// Test the CRON expression
|
||||
if (empty($this->schedule)) {
|
||||
throw new InvalidArgumentException(__('Please enter a CRON expression in the Schedule'), 'schedule');
|
||||
}
|
||||
|
||||
try {
|
||||
$cron = new CronExpression($this->schedule);
|
||||
$cron->getNextRunDate();
|
||||
} catch (\Exception $e) {
|
||||
$this->getLog()->info('run: CRON syntax error for taskId ' . $this->taskId
|
||||
. ', e: ' . $e->getMessage());
|
||||
|
||||
try {
|
||||
$trimmed = substr($this->schedule, 0, strlen($this->schedule) - 2);
|
||||
$cron = new CronExpression($trimmed);
|
||||
$cron->getNextRunDate();
|
||||
} catch (\Exception) {
|
||||
throw new InvalidArgumentException(__('Invalid CRON expression in the Schedule'), 'schedule');
|
||||
}
|
||||
|
||||
// Swap to the trimmed (and correct) schedule
|
||||
$this->schedule = $trimmed;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Save
|
||||
* @throws InvalidArgumentException
|
||||
*/
|
||||
public function save(array $options = []): void
|
||||
{
|
||||
$options = array_merge([
|
||||
'validate' => true,
|
||||
], $options);
|
||||
|
||||
if ($options['validate']) {
|
||||
$this->validate();
|
||||
}
|
||||
|
||||
if ($this->taskId == null) {
|
||||
$this->add();
|
||||
} else {
|
||||
// If we've transitioned from active to inactive, then reset the task status
|
||||
if ($this->getOriginalValue('isActive') != $this->isActive) {
|
||||
$this->status = Task::$STATUS_IDLE;
|
||||
}
|
||||
|
||||
$this->edit();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Delete
|
||||
*/
|
||||
public function delete()
|
||||
{
|
||||
$this->getStore()->update('DELETE FROM `task` WHERE `taskId` = :taskId', ['taskId' => $this->taskId]);
|
||||
}
|
||||
|
||||
private function add()
|
||||
{
|
||||
$this->taskId = $this->getStore()->insert('
|
||||
INSERT INTO `task` (`name`, `status`, `configFile`, `class`, `pid`, `options`, `schedule`,
|
||||
`lastRunDt`, `lastRunMessage`, `lastRunStatus`, `lastRunDuration`, `lastRunExitCode`,
|
||||
`isActive`, `runNow`) VALUES
|
||||
(:name, :status, :configFile, :class, :pid, :options, :schedule,
|
||||
:lastRunDt, :lastRunMessage, :lastRunStatus, :lastRunDuration, :lastRunExitCode,
|
||||
:isActive, :runNow)
|
||||
', [
|
||||
'name' => $this->name,
|
||||
'status' => $this->status,
|
||||
'pid' => $this->pid,
|
||||
'configFile' => $this->configFile,
|
||||
'class' => $this->class,
|
||||
'options' => json_encode($this->options),
|
||||
'schedule' => $this->schedule,
|
||||
'lastRunDt' => $this->lastRunDt,
|
||||
'lastRunMessage' => $this->lastRunMessage,
|
||||
'lastRunStatus' => $this->lastRunStatus,
|
||||
'lastRunDuration' => $this->lastRunDuration,
|
||||
'lastRunExitCode' => $this->lastRunExitCode,
|
||||
'isActive' => $this->isActive,
|
||||
'runNow' => $this->runNow
|
||||
]);
|
||||
}
|
||||
|
||||
private function edit()
|
||||
{
|
||||
$this->getStore()->update('
|
||||
UPDATE `task` SET
|
||||
`name` = :name,
|
||||
`status` = :status,
|
||||
`pid` = :pid,
|
||||
`configFile` = :configFile,
|
||||
`class` = :class,
|
||||
`options` = :options,
|
||||
`schedule` = :schedule,
|
||||
`lastRunDt` = :lastRunDt,
|
||||
`lastRunMessage` = :lastRunMessage,
|
||||
`lastRunStatus` = :lastRunStatus,
|
||||
`lastRunDuration` = :lastRunDuration,
|
||||
`lastRunExitCode` = :lastRunExitCode,
|
||||
`isActive` = :isActive,
|
||||
`runNow` = :runNow
|
||||
WHERE `taskId` = :taskId
|
||||
', [
|
||||
'taskId' => $this->taskId,
|
||||
'name' => $this->name,
|
||||
'status' => $this->status,
|
||||
'pid' => $this->pid,
|
||||
'configFile' => $this->configFile,
|
||||
'class' => $this->class,
|
||||
'options' => json_encode($this->options),
|
||||
'schedule' => $this->schedule,
|
||||
'lastRunDt' => $this->lastRunDt,
|
||||
'lastRunMessage' => $this->lastRunMessage,
|
||||
'lastRunStatus' => $this->lastRunStatus,
|
||||
'lastRunDuration' => $this->lastRunDuration,
|
||||
'lastRunExitCode' => $this->lastRunExitCode,
|
||||
'isActive' => $this->isActive,
|
||||
'runNow' => $this->runNow
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Set this task to be started, updating the DB as necessary
|
||||
* @return $this
|
||||
*/
|
||||
public function setStarted(): Task
|
||||
{
|
||||
// Set to running
|
||||
$this->status = \Xibo\Entity\Task::$STATUS_RUNNING;
|
||||
$this->lastRunStartDt = Carbon::now()->format('U');
|
||||
$this->pid = getmypid();
|
||||
|
||||
$this->store->update('
|
||||
UPDATE `task` SET `status` = :status, lastRunStartDt = :lastRunStartDt, pid = :pid
|
||||
WHERE taskId = :taskId
|
||||
', [
|
||||
'taskId' => $this->taskId,
|
||||
'status' => $this->status,
|
||||
'lastRunStartDt' => $this->lastRunStartDt,
|
||||
'pid' => $this->pid,
|
||||
], 'xtr', true, false);
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set this task to be finished, updating only the fields we might have changed
|
||||
* @return $this
|
||||
*/
|
||||
public function setFinished(): Task
|
||||
{
|
||||
$this->getStore()->update('
|
||||
UPDATE `task` SET
|
||||
`status` = :status,
|
||||
`pid` = :pid,
|
||||
`lastRunDt` = :lastRunDt,
|
||||
`lastRunMessage` = :lastRunMessage,
|
||||
`lastRunStatus` = :lastRunStatus,
|
||||
`lastRunDuration` = :lastRunDuration,
|
||||
`lastRunExitCode` = :lastRunExitCode,
|
||||
`runNow` = :runNow
|
||||
WHERE `taskId` = :taskId
|
||||
', [
|
||||
'taskId' => $this->taskId,
|
||||
'status' => $this->status,
|
||||
'pid' => $this->pid,
|
||||
'lastRunDt' => $this->lastRunDt,
|
||||
'lastRunMessage' => $this->lastRunMessage,
|
||||
'lastRunStatus' => $this->lastRunStatus,
|
||||
'lastRunDuration' => $this->lastRunDuration,
|
||||
'lastRunExitCode' => $this->lastRunExitCode,
|
||||
'runNow' => $this->runNow
|
||||
], 'xtr', true, false);
|
||||
|
||||
return $this;
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user