mirror of
https://github.com/Combodo/iTop.git
synced 2026-02-13 07:24:13 +01:00
N°3234 - Cron rework
* some timezone refactors occurred
This commit is contained in:
@@ -769,19 +769,22 @@ class utils
|
|||||||
return new Config();
|
return new Config();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function InitTimeZone() {
|
public static function InitTimeZone($oConfig = null)
|
||||||
$oConfig = self::GetConfig();
|
{
|
||||||
|
if (is_null($oConfig))
|
||||||
|
{
|
||||||
|
$oConfig = self::GetConfig();
|
||||||
|
}
|
||||||
$sItopTimeZone = $oConfig->Get('timezone');
|
$sItopTimeZone = $oConfig->Get('timezone');
|
||||||
|
|
||||||
if (!empty($sItopTimeZone))
|
if (!empty($sItopTimeZone))
|
||||||
{
|
{
|
||||||
date_default_timezone_set($sItopTimeZone);
|
date_default_timezone_set($sItopTimeZone);
|
||||||
}
|
}
|
||||||
else
|
// else
|
||||||
{
|
// {
|
||||||
// Leave as is... up to the admin to set a value somewhere...
|
// // Leave as is... up to the admin to set a value somewhere...
|
||||||
// see http://php.net/manual/en/datetime.configuration.php#ini.date.timezone
|
// // see http://php.net/manual/en/datetime.configuration.php#ini.date.timezone
|
||||||
}
|
// }
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -185,12 +185,9 @@ abstract class AbstractWeeklyScheduledProcess implements iScheduledProcess
|
|||||||
static::DEFAULT_MODULE_SETTING_ENABLED
|
static::DEFAULT_MODULE_SETTING_ENABLED
|
||||||
);
|
);
|
||||||
|
|
||||||
$sItopTimeZone = $this->getOConfig()->Get('timezone');
|
|
||||||
$timezone = new DateTimeZone($sItopTimeZone);
|
|
||||||
|
|
||||||
if (!$bEnabled)
|
if (!$bEnabled)
|
||||||
{
|
{
|
||||||
return new DateTime('3000-01-01', $timezone);
|
return new DateTime('3000-01-01');
|
||||||
}
|
}
|
||||||
|
|
||||||
// 1st - Interpret the list of days as ordered numbers (monday = 1)
|
// 1st - Interpret the list of days as ordered numbers (monday = 1)
|
||||||
@@ -209,7 +206,7 @@ abstract class AbstractWeeklyScheduledProcess implements iScheduledProcess
|
|||||||
throw new ProcessInvalidConfigException($this->GetModuleName().": wrong format for setting '".static::MODULE_SETTING_TIME."' (found '$sProcessTime')");
|
throw new ProcessInvalidConfigException($this->GetModuleName().": wrong format for setting '".static::MODULE_SETTING_TIME."' (found '$sProcessTime')");
|
||||||
}
|
}
|
||||||
|
|
||||||
$oNow = new DateTime($sCurrentTime, $timezone);
|
$oNow = new DateTime($sCurrentTime);
|
||||||
$iNextPos = false;
|
$iNextPos = false;
|
||||||
$sDay = $oNow->format('N');
|
$sDay = $oNow->format('N');
|
||||||
for ($iDay = (int) $sDay; $iDay <= 7; $iDay++)
|
for ($iDay = (int) $sDay; $iDay <= 7; $iDay++)
|
||||||
@@ -244,7 +241,6 @@ abstract class AbstractWeeklyScheduledProcess implements iScheduledProcess
|
|||||||
$oRet->modify('+'.$iMove.' days');
|
$oRet->modify('+'.$iMove.' days');
|
||||||
}
|
}
|
||||||
list($sHours, $sMinutes) = explode(':', $sProcessTime);
|
list($sHours, $sMinutes) = explode(':', $sProcessTime);
|
||||||
/** @noinspection PhpElementIsNotAvailableInCurrentPhpVersionInspection non used new parameter in PHP 7.1 */
|
|
||||||
$oRet->setTime((int)$sHours, (int)$sMinutes);
|
$oRet->setTime((int)$sHours, (int)$sMinutes);
|
||||||
|
|
||||||
return $oRet;
|
return $oRet;
|
||||||
|
|||||||
@@ -54,7 +54,7 @@ class BackgroundTask extends DBObject
|
|||||||
MetaModel::Init_AddAttribute(new AttributeDecimal("average_run_duration", array("allowed_values"=>null, "sql"=>"average_run_duration", "digits"=> 8, "decimals"=> 3, "default_value"=>"0", "is_null_allowed"=>true, "depends_on"=>array())));
|
MetaModel::Init_AddAttribute(new AttributeDecimal("average_run_duration", array("allowed_values"=>null, "sql"=>"average_run_duration", "digits"=> 8, "decimals"=> 3, "default_value"=>"0", "is_null_allowed"=>true, "depends_on"=>array())));
|
||||||
|
|
||||||
MetaModel::Init_AddAttribute(new AttributeBoolean("running", array("allowed_values"=>null, "sql"=>"running", "default_value"=>false, "is_null_allowed"=>false, "depends_on"=>array())));
|
MetaModel::Init_AddAttribute(new AttributeBoolean("running", array("allowed_values"=>null, "sql"=>"running", "default_value"=>false, "is_null_allowed"=>false, "depends_on"=>array())));
|
||||||
MetaModel::Init_AddAttribute(new AttributeEnum("status", array("allowed_values"=>new ValueSetEnum('active,paused'), "sql"=>"status", "default_value"=>'active', "is_null_allowed"=>false, "depends_on"=>array())));
|
MetaModel::Init_AddAttribute(new AttributeEnum("status", array("allowed_values"=>new ValueSetEnum('active,paused,removed'), "sql"=>"status", "default_value"=>'active', "is_null_allowed"=>false, "depends_on"=>array())));
|
||||||
MetaModel::Init_AddAttribute(new AttributeString("system_user", array("allowed_values"=>null, "sql"=>"system_user", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
|
MetaModel::Init_AddAttribute(new AttributeString("system_user", array("allowed_values"=>null, "sql"=>"system_user", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -129,8 +129,7 @@ abstract class RotatingLogFileNameBuilder implements iLogFileNameBuilder
|
|||||||
public function CheckAndRotateLogFile()
|
public function CheckAndRotateLogFile()
|
||||||
{
|
{
|
||||||
$oConfig = utils::GetConfig();
|
$oConfig = utils::GetConfig();
|
||||||
$sItopTimeZone = $oConfig->Get('timezone');
|
utils::InitTimeZone($oConfig);
|
||||||
$timezone = new DateTimeZone($sItopTimeZone);
|
|
||||||
|
|
||||||
if ($this->GetLastModifiedDateForFile() === null)
|
if ($this->GetLastModifiedDateForFile() === null)
|
||||||
{
|
{
|
||||||
@@ -145,11 +144,13 @@ abstract class RotatingLogFileNameBuilder implements iLogFileNameBuilder
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
$oDateTime = DateTime::createFromFormat('U', $iLogDateLastModifiedTimeStamp);
|
$oDateTime = DateTime::createFromFormat('U', $iLogDateLastModifiedTimeStamp);
|
||||||
|
$sItopTimeZone = $oConfig->Get('timezone');
|
||||||
|
$timezone = new DateTimeZone($sItopTimeZone);
|
||||||
$oDateTime->setTimezone($timezone);
|
$oDateTime->setTimezone($timezone);
|
||||||
$this->SetLastModifiedDateForFile($oDateTime);
|
$this->SetLastModifiedDateForFile($oDateTime);
|
||||||
}
|
}
|
||||||
|
|
||||||
$oNow = new DateTime('now', $timezone);
|
$oNow = new DateTime();
|
||||||
$bShouldRotate = $this->ShouldRotate($this->GetLastModifiedDateForFile(), $oNow);
|
$bShouldRotate = $this->ShouldRotate($this->GetLastModifiedDateForFile(), $oNow);
|
||||||
if (!$bShouldRotate)
|
if (!$bShouldRotate)
|
||||||
{
|
{
|
||||||
@@ -830,4 +831,4 @@ class LogFileRotationProcess implements iScheduledProcess
|
|||||||
|
|
||||||
throw new ProcessException(self::class.' : The configured filename builder is invalid (log_filename_builder_impl="'.$sLogFileNameBuilder.'")');
|
throw new ProcessException(self::class.' : The configured filename builder is invalid (log_filename_builder_impl="'.$sLogFileNameBuilder.'")');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6304,6 +6304,7 @@ abstract class MetaModel
|
|||||||
|
|
||||||
// N°2478 utils has his own private attribute
|
// N°2478 utils has his own private attribute
|
||||||
// @see utils::GetConfig : it always call MetaModel, but to be sure we're doing this extra copy anyway O:)
|
// @see utils::GetConfig : it always call MetaModel, but to be sure we're doing this extra copy anyway O:)
|
||||||
|
utils::InitTimeZone($oConfiguration);
|
||||||
utils::SetConfig($oConfiguration);
|
utils::SetConfig($oConfiguration);
|
||||||
|
|
||||||
// Set log ASAP
|
// Set log ASAP
|
||||||
@@ -6344,19 +6345,6 @@ abstract class MetaModel
|
|||||||
DBSearch::EnableQueryIndentation(self::$m_oConfig->Get('query_indentation_enabled'));
|
DBSearch::EnableQueryIndentation(self::$m_oConfig->Get('query_indentation_enabled'));
|
||||||
DBSearch::EnableOptimizeQuery(self::$m_oConfig->Get('query_optimization_enabled'));
|
DBSearch::EnableOptimizeQuery(self::$m_oConfig->Get('query_optimization_enabled'));
|
||||||
|
|
||||||
// PHP timezone first...
|
|
||||||
//
|
|
||||||
$sPHPTimezone = self::$m_oConfig->Get('timezone');
|
|
||||||
if ($sPHPTimezone == '')
|
|
||||||
{
|
|
||||||
// Leave as is... up to the admin to set a value somewhere...
|
|
||||||
//$sPHPTimezone = date_default_timezone_get();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
date_default_timezone_set($sPHPTimezone);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Note: load the dictionary as soon as possible, because it might be
|
// Note: load the dictionary as soon as possible, because it might be
|
||||||
// needed when some error occur
|
// needed when some error occur
|
||||||
$sAppIdentity = 'itop-'.MetaModel::GetEnvironmentId();
|
$sAppIdentity = 'itop-'.MetaModel::GetEnvironmentId();
|
||||||
|
|||||||
@@ -6,8 +6,6 @@ namespace Combodo\iTop\Test\UnitTest\Core;
|
|||||||
|
|
||||||
use Combodo\iTop\Test\UnitTest\ItopTestCase;
|
use Combodo\iTop\Test\UnitTest\ItopTestCase;
|
||||||
use DailyRotatingLogFileNameBuilder;
|
use DailyRotatingLogFileNameBuilder;
|
||||||
use WeeklyRotatingLogFileNameBuilder;
|
|
||||||
use MonthlyRotatingLogFileNameBuilder;
|
|
||||||
use DateTime;
|
use DateTime;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -90,7 +88,7 @@ class LogFileNameBuilderTest extends ItopTestCase
|
|||||||
$oFileBuilder->ResetLastModifiedDateForFile();
|
$oFileBuilder->ResetLastModifiedDateForFile();
|
||||||
$oFileBuilder->CheckAndRotateLogFile();
|
$oFileBuilder->CheckAndRotateLogFile();
|
||||||
$this->assertFileNotExists($sLogFile, 'Test log file modification date is yesterday, file rotated !');
|
$this->assertFileNotExists($sLogFile, 'Test log file modification date is yesterday, file rotated !');
|
||||||
$this->assertFileExists($sRotatedLogFile, 'Rotation was done');
|
$this->assertFileExists($sRotatedLogFile, 'Rotation was not done');
|
||||||
|
|
||||||
// file cleanup will be done in tearDown !
|
// file cleanup will be done in tearDown !
|
||||||
}
|
}
|
||||||
@@ -165,4 +163,4 @@ class LogFileNameBuilderTest extends ItopTestCase
|
|||||||
|
|
||||||
$this->assertEquals($sExpectedOccurrence, $sActualOccurrence);
|
$this->assertEquals($sExpectedOccurrence, $sActualOccurrence);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ class WeeklyScheduledProcessMockConfig extends AbstractWeeklyScheduledProcess
|
|||||||
$this->oConfig->SetModuleSetting(self::MODULE_NAME, self::MODULE_SETTING_ENABLED, $bEnabledValue);
|
$this->oConfig->SetModuleSetting(self::MODULE_NAME, self::MODULE_SETTING_ENABLED, $bEnabledValue);
|
||||||
$this->oConfig->SetModuleSetting(self::MODULE_NAME, self::MODULE_SETTING_TIME, $sTimeValue);
|
$this->oConfig->SetModuleSetting(self::MODULE_NAME, self::MODULE_SETTING_TIME, $sTimeValue);
|
||||||
$this->oConfig->SetModuleSetting(self::MODULE_NAME, self::MODULE_SETTING_WEEKDAYS, 'monday, tuesday, wednesday, thursday, friday');
|
$this->oConfig->SetModuleSetting(self::MODULE_NAME, self::MODULE_SETTING_WEEKDAYS, 'monday, tuesday, wednesday, thursday, friday');
|
||||||
|
utils::InitTimeZone($this->oConfig);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function GetModuleName()
|
protected function GetModuleName()
|
||||||
@@ -33,4 +34,4 @@ class WeeklyScheduledProcessMockConfig extends AbstractWeeklyScheduledProcess
|
|||||||
{
|
{
|
||||||
// nothing to do here (not tested)
|
// nothing to do here (not tested)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,6 @@
|
|||||||
namespace Combodo\iTop\Test\UnitTest\Core;
|
namespace Combodo\iTop\Test\UnitTest\Core;
|
||||||
|
|
||||||
use Combodo\iTop\Test\UnitTest\ItopTestCase;
|
use Combodo\iTop\Test\UnitTest\ItopTestCase;
|
||||||
use Config;
|
|
||||||
use DateTime;
|
use DateTime;
|
||||||
|
|
||||||
|
|
||||||
@@ -38,9 +37,7 @@ class WeeklyScheduledProcessTest extends ItopTestCase
|
|||||||
{
|
{
|
||||||
$oWeeklyImpl = new \WeeklyScheduledProcessMockConfig($bEnabledValue, $sTimeValue);
|
$oWeeklyImpl = new \WeeklyScheduledProcessMockConfig($bEnabledValue, $sTimeValue);
|
||||||
|
|
||||||
$sItopTimeZone = $oWeeklyImpl->getOConfig()->Get('timezone');
|
$oExpectedDateTime = new DateTime($sExpectedTime);
|
||||||
$timezone = new \DateTimeZone($sItopTimeZone);
|
|
||||||
$oExpectedDateTime = new DateTime($sExpectedTime, $timezone);
|
|
||||||
|
|
||||||
$this->assertEquals($oExpectedDateTime, $oWeeklyImpl->GetNextOccurrence($sCurrentTime));
|
$this->assertEquals($oExpectedDateTime, $oWeeklyImpl->GetNextOccurrence($sCurrentTime));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -48,7 +48,7 @@ $oCtx = new ContextTag(ContextTag::TAG_CRON);
|
|||||||
|
|
||||||
function ReadMandatoryParam($oP, $sParam, $sSanitizationFilter = 'parameter')
|
function ReadMandatoryParam($oP, $sParam, $sSanitizationFilter = 'parameter')
|
||||||
{
|
{
|
||||||
$sValue = utils::ReadParam($sParam, null, true /* Allow CLI */, $sSanitizationFilter);
|
$sValue = utils::ReadParam($sParam, null, true, $sSanitizationFilter);
|
||||||
if (is_null($sValue))
|
if (is_null($sValue))
|
||||||
{
|
{
|
||||||
$oP->p("ERROR: Missing argument '$sParam'\n");
|
$oP->p("ERROR: Missing argument '$sParam'\n");
|
||||||
@@ -76,22 +76,29 @@ function UsageAndExit($oP)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param iProcess $oProcess
|
|
||||||
* @param \BackgroundTask $oTask
|
* @param \BackgroundTask $oTask
|
||||||
* @param DateTime $oStartDate
|
|
||||||
* @param int $iTimeLimit
|
* @param int $iTimeLimit
|
||||||
*
|
*
|
||||||
* @return string
|
* @return string
|
||||||
|
* @throws \ArchivedObjectException
|
||||||
|
* @throws \CoreCannotSaveObjectException
|
||||||
|
* @throws \CoreException
|
||||||
|
* @throws \CoreUnexpectedValue
|
||||||
|
* @throws \MySQLHasGoneAwayException
|
||||||
* @throws \ProcessFatalException
|
* @throws \ProcessFatalException
|
||||||
* @throws MySQLHasGoneAwayException
|
* @throws \ReflectionException
|
||||||
|
* @throws \Exception
|
||||||
*/
|
*/
|
||||||
function RunTask($oProcess, BackgroundTask $oTask, $oStartDate, $iTimeLimit)
|
function RunTask(BackgroundTask $oTask, $iTimeLimit)
|
||||||
{
|
{
|
||||||
|
$TaskClass = $oTask->Get('class_name');
|
||||||
|
$oProcess = new $TaskClass;
|
||||||
$oDateStarted = new DateTime();
|
$oDateStarted = new DateTime();
|
||||||
|
$oDatePlanned = new DateTime($oTask->Get('next_run_date'));
|
||||||
$fStart = microtime(true);
|
$fStart = microtime(true);
|
||||||
$oCtx = new ContextTag('CRON:Task:'.$oTask->Get('class_name'));
|
$oCtx = new ContextTag('CRON:Task:'.$TaskClass);
|
||||||
|
|
||||||
$sMessage = "";
|
$sMessage = '';
|
||||||
$oExceptionToThrow = null;
|
$oExceptionToThrow = null;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -135,15 +142,14 @@ function RunTask($oProcess, BackgroundTask $oTask, $oStartDate, $iTimeLimit)
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Background processes do repeat periodically
|
// Background processes do repeat periodically
|
||||||
$oPlannedStart = clone $oDateStarted;
|
$oPlannedStart = clone $oDatePlanned;
|
||||||
// Let's assume that the task was started exactly when planned so that the schedule does no shift each time
|
// Let's schedule from the previous planned date of execution to avoid shift
|
||||||
// this allows to schedule a task everyday "around" 11:30PM for example
|
|
||||||
$oPlannedStart->modify('+'.$oProcess->GetPeriodicity().' seconds');
|
$oPlannedStart->modify('+'.$oProcess->GetPeriodicity().' seconds');
|
||||||
$oEnd = new DateTime();
|
$oEnd = new DateTime();
|
||||||
if ($oPlannedStart->format('U') < $oEnd->format('U'))
|
while ($oPlannedStart->format('U') < $oEnd->format('U'))
|
||||||
{
|
{
|
||||||
// Huh, next planned start is already in the past, shift it of the periodicity !
|
// Next planned start is already in the past, increase it again by a period
|
||||||
$oPlannedStart = $oEnd->modify('+'.$oProcess->GetPeriodicity().' seconds');
|
$oPlannedStart = $oPlannedStart->modify('+'.$oProcess->GetPeriodicity().' seconds');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -162,67 +168,32 @@ function RunTask($oProcess, BackgroundTask $oTask, $oStartDate, $iTimeLimit)
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @param CLIPage|WebPage $oP
|
* @param CLIPage|WebPage $oP
|
||||||
* @param iProcess[] $aProcesses
|
|
||||||
* @param boolean $bVerbose
|
* @param boolean $bVerbose
|
||||||
|
*
|
||||||
|
* @throws \ArchivedObjectException
|
||||||
|
* @throws \CoreCannotSaveObjectException
|
||||||
|
* @throws \CoreException
|
||||||
|
* @throws \CoreUnexpectedValue
|
||||||
|
* @throws \MissingQueryArgument
|
||||||
|
* @throws \MySQLException
|
||||||
|
* @throws \MySQLHasGoneAwayException
|
||||||
|
* @throws \ReflectionException
|
||||||
*/
|
*/
|
||||||
function CronExec($oP, $aProcesses, $bVerbose)
|
function CronExec($oP, $bVerbose)
|
||||||
{
|
{
|
||||||
$iStarted = time();
|
$iStarted = time();
|
||||||
$iMaxDuration = MetaModel::GetConfig()->Get('cron_max_execution_time');
|
$iMaxDuration = MetaModel::GetConfig()->Get('cron_max_execution_time');
|
||||||
$iTimeLimit = $iStarted + $iMaxDuration;
|
$iTimeLimit = $iStarted + $iMaxDuration;
|
||||||
|
$iCronSleep = MetaModel::GetConfig()->Get('cron_sleep');
|
||||||
|
|
||||||
if ($bVerbose)
|
if ($bVerbose)
|
||||||
{
|
{
|
||||||
$oP->p("Planned duration = $iMaxDuration seconds");
|
$oP->p("Planned duration = $iMaxDuration seconds");
|
||||||
|
$oP->p("Loop pause = $iCronSleep seconds");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Reset the next planned execution to take into account new settings
|
ReSyncProcesses($oP, $bVerbose);
|
||||||
$oSearch = new DBObjectSearch('BackgroundTask');
|
|
||||||
/** @var DBObjectSet $oTasks */
|
|
||||||
$oTasks = new DBObjectSet($oSearch);
|
|
||||||
/** @var BackgroundTask $oTask */
|
|
||||||
while ($oTask = $oTasks->Fetch())
|
|
||||||
{
|
|
||||||
$sTaskClass = $oTask->Get('class_name');
|
|
||||||
// The BackgroundTask can point to a non existing class : this could happen for example if an extension has been removed
|
|
||||||
// we could also try/catch when instanciating ReflectionClass, but sometimes old recipes are good too ;)
|
|
||||||
if (!class_exists($sTaskClass))
|
|
||||||
{
|
|
||||||
if ($oTask->Get('status') == 'active')
|
|
||||||
{
|
|
||||||
$oP->p("ERROR : the background task was paused because it references the non existing class '$sTaskClass'");
|
|
||||||
|
|
||||||
$oTask->Set('status', 'paused');
|
|
||||||
$oTask->DBUpdate();
|
|
||||||
}
|
|
||||||
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$oRefClass = new ReflectionClass($sTaskClass);
|
|
||||||
if (!$oRefClass->implementsInterface('iScheduledProcess'))
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$oNow = new DateTime();
|
|
||||||
if (($oTask->Get('status') != 'active')
|
|
||||||
|| ($oTask->Get('next_run_date') > $oNow->format('Y-m-d H:i:s')))
|
|
||||||
{
|
|
||||||
if ($bVerbose)
|
|
||||||
{
|
|
||||||
$oP->p("Resetting the next run date for $sTaskClass");
|
|
||||||
}
|
|
||||||
$oProcess = $aProcesses[$sTaskClass];
|
|
||||||
$oNextOcc = $oProcess->GetNextOccurrence();
|
|
||||||
$oTask->Set('next_run_date', $oNextOcc->format('Y-m-d H:i:s'));
|
|
||||||
$oTask->DBUpdate();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$iCronSleep = MetaModel::GetConfig()->Get('cron_sleep');
|
|
||||||
|
|
||||||
$oSearch = new DBObjectSearch('BackgroundTask');
|
|
||||||
while (time() < $iTimeLimit)
|
while (time() < $iTimeLimit)
|
||||||
{
|
{
|
||||||
// Verify files instead of reloading the full config each time
|
// Verify files instead of reloading the full config each time
|
||||||
@@ -232,31 +203,191 @@ function CronExec($oP, $aProcesses, $bVerbose)
|
|||||||
exit(EXIT_CODE_ERROR);
|
exit(EXIT_CODE_ERROR);
|
||||||
}
|
}
|
||||||
|
|
||||||
$oTasks = new DBObjectSet($oSearch);
|
$oNow = new DateTime();
|
||||||
$aTasks = array();
|
$sNow = $oNow->format('Y-m-d H:i:s');
|
||||||
while ($oTask = $oTasks->Fetch())
|
$oSearch = new DBObjectSearch('BackgroundTask');
|
||||||
|
$oSearch->AddCondition('next_run_date', $sNow, '<=');
|
||||||
|
$oSearch->AddCondition('status', 'active');
|
||||||
|
$oTasks = new DBObjectSet($oSearch, ['next_run_date' => true]);
|
||||||
|
|
||||||
|
if ($oTasks->CountExceeds(0))
|
||||||
{
|
{
|
||||||
$aTasks[$oTask->Get('class_name')] = $oTask;
|
$aTasks = array();
|
||||||
|
if ($bVerbose)
|
||||||
|
{
|
||||||
|
$oP->p("Tasks planned to run now ($sNow):");
|
||||||
|
$oP->p('+---------------------------+---------+---------------------+---------------------+');
|
||||||
|
$oP->p('| Task Class | Status | Last Run | Next Run |');
|
||||||
|
$oP->p('+---------------------------+---------+---------------------+---------------------+');
|
||||||
|
}
|
||||||
|
while ($oTask = $oTasks->Fetch())
|
||||||
|
{
|
||||||
|
$aTasks[$oTask->Get('class_name')] = $oTask;
|
||||||
|
if ($bVerbose)
|
||||||
|
{
|
||||||
|
$sTaskName = $oTask->Get('class_name');
|
||||||
|
$sStatus = $oTask->Get('status');
|
||||||
|
$sLastRunDate = $oTask->Get('latest_run_date');
|
||||||
|
$sNextRunDate = $oTask->Get('next_run_date');
|
||||||
|
$oP->p(sprintf('| %1$-25.25s | %2$-7s | %3$-19s | %4$-19s |', $sTaskName, $sStatus, $sLastRunDate, $sNextRunDate));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ($bVerbose)
|
||||||
|
{
|
||||||
|
$oP->p('+---------------------------+---------+---------------------+---------------------+');
|
||||||
|
}
|
||||||
|
$aRunTasks = [];
|
||||||
|
foreach ($aTasks as $oTask)
|
||||||
|
{
|
||||||
|
$sTaskClass = $oTask->Get('class_name');
|
||||||
|
$aRunTasks[] = $sTaskClass;
|
||||||
|
|
||||||
|
// N°3219 for each process will use a specific CMDBChange object with a specific track info
|
||||||
|
// Any BackgroundProcess can overrides this as needed
|
||||||
|
CMDBObject::SetCurrentChange(null);
|
||||||
|
CMDBObject::SetTrackInfo("Background task ($sTaskClass)");
|
||||||
|
CMDBObject::SetTrackOrigin(null);
|
||||||
|
|
||||||
|
// Run the task and record its next run time
|
||||||
|
if ($bVerbose)
|
||||||
|
{
|
||||||
|
$oP->p(">> === ".$oNow->format('Y-m-d H:i:s').sprintf(" Starting:%-'=49s", ' '.$sTaskClass.' '));
|
||||||
|
}
|
||||||
|
try
|
||||||
|
{
|
||||||
|
$sMessage = RunTask($aTasks[$sTaskClass], $iTimeLimit);
|
||||||
|
} catch (MySQLHasGoneAwayException $e)
|
||||||
|
{
|
||||||
|
$oP->p("ERROR : 'MySQL has gone away' thrown when processing $sTaskClass (error_code=".$e->getCode().")");
|
||||||
|
exit(EXIT_CODE_FATAL);
|
||||||
|
} catch (ProcessFatalException $e)
|
||||||
|
{
|
||||||
|
$oP->p("ERROR : an exception was thrown when processing '$sTaskClass' (".$e->getInfoLog().")");
|
||||||
|
IssueLog::Error("Cron.php error : an exception was thrown when processing '$sTaskClass' (".$e->getInfoLog().')');
|
||||||
|
}
|
||||||
|
if ($bVerbose)
|
||||||
|
{
|
||||||
|
if (!empty($sMessage))
|
||||||
|
{
|
||||||
|
$oP->p("$sTaskClass: $sMessage");
|
||||||
|
}
|
||||||
|
$oEnd = new DateTime();
|
||||||
|
$sNextRunDate = $oTask->Get('next_run_date');
|
||||||
|
$oP->p("<< === ".$oEnd->format('Y-m-d H:i:s').sprintf(" End of: %-'=40s", ' '.$sTaskClass.' ')." Next: $sNextRunDate");
|
||||||
|
}
|
||||||
|
if (time() > $iTimeLimit)
|
||||||
|
{
|
||||||
|
break 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tasks to run later
|
||||||
|
if ($bVerbose)
|
||||||
|
{
|
||||||
|
$oP->p('');
|
||||||
|
$oSearch = new DBObjectSearch('BackgroundTask');
|
||||||
|
$oSearch->AddCondition('next_run_date', $sNow, '>');
|
||||||
|
$oSearch->AddCondition('status', 'active');
|
||||||
|
$oTasks = new DBObjectSet($oSearch, ['next_run_date' => true]);
|
||||||
|
while ($oTask = $oTasks->Fetch())
|
||||||
|
{
|
||||||
|
if (!in_array($oTask->Get('class_name'), $aRunTasks))
|
||||||
|
{
|
||||||
|
$oP->p(sprintf("-- Skipping task: %-'-40s", $oTask->Get('class_name').' ')." until: ".$oTask->Get('next_run_date'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$oNow = new DateTime();
|
if ($bVerbose)
|
||||||
ReorderProcesses($aProcesses, $aTasks, $oNow, $bVerbose, $oP);
|
|
||||||
|
|
||||||
foreach ($aProcesses as $oProcess)
|
|
||||||
{
|
{
|
||||||
$sTaskClass = get_class($oProcess);
|
$oP->p("Sleeping $iCronSleep s\n");
|
||||||
|
}
|
||||||
|
sleep($iCronSleep);
|
||||||
|
}
|
||||||
|
if ($bVerbose)
|
||||||
|
{
|
||||||
|
$oP->p('');
|
||||||
|
DisplayStatus($oP, ['next_run_date' => true]);
|
||||||
|
$oP->p("Reached normal execution time limit (exceeded by ".(time() - $iTimeLimit)."s)");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// N°3219 for each process will use a specific CMDBChange object with a specific track info
|
/**
|
||||||
// Any BackgroundProcess can overrides this as needed
|
* @param $oP
|
||||||
CMDBObject::SetCurrentChange(null);
|
* @param array $aOrderBy
|
||||||
CMDBObject::SetTrackInfo("Background task ($sTaskClass)");
|
*
|
||||||
CMDBObject::SetTrackOrigin(null);
|
* @throws \ArchivedObjectException
|
||||||
|
* @throws \CoreException
|
||||||
|
* @throws \CoreUnexpectedValue
|
||||||
|
* @throws \MySQLException
|
||||||
|
*/
|
||||||
|
function DisplayStatus($oP, $aOrderBy = [])
|
||||||
|
{
|
||||||
|
$oSearch = new DBObjectSearch('BackgroundTask');
|
||||||
|
$oTasks = new DBObjectSet($oSearch, $aOrderBy);
|
||||||
|
$oP->p('+---------------------------+---------+---------------------+---------------------+--------+-----------+');
|
||||||
|
$oP->p('| Task Class | Status | Last Run | Next Run | Nb Run | Avg. Dur. |');
|
||||||
|
$oP->p('+---------------------------+---------+---------------------+---------------------+--------+-----------+');
|
||||||
|
while ($oTask = $oTasks->Fetch())
|
||||||
|
{
|
||||||
|
$sTaskName = $oTask->Get('class_name');
|
||||||
|
$sStatus = $oTask->Get('status');
|
||||||
|
$sLastRunDate = $oTask->Get('latest_run_date');
|
||||||
|
$sNextRunDate = $oTask->Get('next_run_date');
|
||||||
|
$iNbRun = (int)$oTask->Get('total_exec_count');
|
||||||
|
$sAverageRunTime = $oTask->Get('average_run_duration');
|
||||||
|
$oP->p(sprintf('| %1$-25.25s | %2$-7s | %3$-19s | %4$-19s | %5$6d | %6$7s s |', $sTaskName, $sStatus,
|
||||||
|
$sLastRunDate, $sNextRunDate, $iNbRun, $sAverageRunTime));
|
||||||
|
}
|
||||||
|
$oP->p('+---------------------------+---------+---------------------+---------------------+--------+-----------+');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param $oP
|
||||||
|
* @param $bVerbose
|
||||||
|
*
|
||||||
|
* @throws \ArchivedObjectException
|
||||||
|
* @throws \CoreCannotSaveObjectException
|
||||||
|
* @throws \CoreException
|
||||||
|
* @throws \CoreUnexpectedValue
|
||||||
|
* @throws \CoreWarning
|
||||||
|
* @throws \MySQLException
|
||||||
|
* @throws \OQLException
|
||||||
|
* @throws \ReflectionException
|
||||||
|
*/
|
||||||
|
function ReSyncProcesses($oP, $bVerbose)
|
||||||
|
{
|
||||||
|
// Enumerate classes implementing BackgroundProcess
|
||||||
|
//
|
||||||
|
$oSearch = new DBObjectSearch('BackgroundTask');
|
||||||
|
$oTasks = new DBObjectSet($oSearch);
|
||||||
|
$aTasks = array();
|
||||||
|
while ($oTask = $oTasks->Fetch())
|
||||||
|
{
|
||||||
|
$aTasks[$oTask->Get('class_name')] = $oTask;
|
||||||
|
}
|
||||||
|
$oNow = new DateTime();
|
||||||
|
|
||||||
|
$aProcesses = array();
|
||||||
|
foreach (get_declared_classes() as $sTaskClass)
|
||||||
|
{
|
||||||
|
$oRefClass = new ReflectionClass($sTaskClass);
|
||||||
|
if ($oRefClass->isAbstract())
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if ($oRefClass->implementsInterface('iProcess'))
|
||||||
|
{
|
||||||
|
$oProcess = new $sTaskClass;
|
||||||
|
$aProcesses[$sTaskClass] = $oProcess;
|
||||||
|
|
||||||
|
// Create missing entry if needed
|
||||||
if (!array_key_exists($sTaskClass, $aTasks))
|
if (!array_key_exists($sTaskClass, $aTasks))
|
||||||
{
|
{
|
||||||
// New entry, let's create a new BackgroundTask record, and plan the first execution
|
// New entry, let's create a new BackgroundTask record, and plan the first execution
|
||||||
$oTask = new BackgroundTask();
|
$oTask = new BackgroundTask();
|
||||||
$oTask->Set('class_name', get_class($oProcess));
|
$oTask->Set('class_name', $sTaskClass);
|
||||||
$oTask->Set('total_exec_count', 0);
|
$oTask->Set('total_exec_count', 0);
|
||||||
$oTask->Set('min_run_duration', 99999.999);
|
$oTask->Set('min_run_duration', 99999.999);
|
||||||
$oTask->Set('max_run_duration', 0);
|
$oTask->Set('max_run_duration', 0);
|
||||||
@@ -278,202 +409,55 @@ function CronExec($oP, $aProcesses, $bVerbose)
|
|||||||
$oP->p('First execution planned at: '.$oTask->Get('next_run_date'));
|
$oP->p('First execution planned at: '.$oTask->Get('next_run_date'));
|
||||||
}
|
}
|
||||||
$oTask->DBInsert();
|
$oTask->DBInsert();
|
||||||
$aTasks[$oTask->Get('class_name')] = $oTask;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (($aTasks[$sTaskClass]->Get('status') == 'active') && ($aTasks[$sTaskClass]->Get('next_run_date') <= $oNow->format('Y-m-d H:i:s')))
|
|
||||||
{
|
|
||||||
// Run the task and record its next run time
|
|
||||||
if ($bVerbose)
|
|
||||||
{
|
|
||||||
$oP->p(">> === ".$oNow->format('Y-m-d H:i:s').sprintf(" Starting:%-'=40s", ' '.$sTaskClass.' '));
|
|
||||||
}
|
|
||||||
try
|
|
||||||
{
|
|
||||||
$sMessage = RunTask($oProcess, $aTasks[$sTaskClass], $oNow, $iTimeLimit);
|
|
||||||
} catch (MySQLHasGoneAwayException $e)
|
|
||||||
{
|
|
||||||
$oP->p("ERROR : 'MySQL has gone away' thrown when processing $sTaskClass (error_code=".$e->getCode().")");
|
|
||||||
exit(EXIT_CODE_FATAL);
|
|
||||||
} catch (ProcessFatalException $e)
|
|
||||||
{
|
|
||||||
$oP->p("ERROR : an exception was thrown when processing '$sTaskClass' (".$e->getInfoLog().")");
|
|
||||||
IssueLog::Error("Cron.php error : an exception was thrown when processing '$sTaskClass' (".$e->getInfoLog().')');
|
|
||||||
}
|
|
||||||
if ($bVerbose)
|
|
||||||
{
|
|
||||||
if (!empty($sMessage))
|
|
||||||
{
|
|
||||||
$oP->p("$sTaskClass: $sMessage");
|
|
||||||
}
|
|
||||||
$oEnd = new DateTime();
|
|
||||||
$oP->p("<< === ".$oEnd->format('Y-m-d H:i:s').sprintf(" End of: %-'=40s", ' '.$sTaskClass.' '));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// will run later
|
/** @var \BackgroundTask $oTask */
|
||||||
if (($aTasks[$sTaskClass]->Get('status') == 'active') && $bVerbose)
|
|
||||||
{
|
|
||||||
$oP->p("Skipping asynchronous task: $sTaskClass until ".$aTasks[$sTaskClass]->Get('next_run_date'));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if ($bVerbose)
|
|
||||||
{
|
|
||||||
$oP->p("Sleeping");
|
|
||||||
}
|
|
||||||
sleep($iCronSleep);
|
|
||||||
}
|
|
||||||
if ($bVerbose)
|
|
||||||
{
|
|
||||||
$oP->p("Reached normal execution time limit (exceeded by ".(time() - $iTimeLimit)."s)");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function DisplayStatus($oP)
|
|
||||||
{
|
|
||||||
$oSearch = new DBObjectSearch('BackgroundTask');
|
|
||||||
$oTasks = new DBObjectSet($oSearch);
|
|
||||||
$oP->p('+---------------------------+---------+---------------------+---------------------+--------+-----------+');
|
|
||||||
$oP->p('| Task Class | Status | Last Run | Next Run | Nb Run | Avg. Dur. |');
|
|
||||||
$oP->p('+---------------------------+---------+---------------------+---------------------+--------+-----------+');
|
|
||||||
while ($oTask = $oTasks->Fetch())
|
|
||||||
{
|
|
||||||
$sTaskName = $oTask->Get('class_name');
|
|
||||||
$sStatus = $oTask->Get('status');
|
|
||||||
$sLastRunDate = $oTask->Get('latest_run_date');
|
|
||||||
$sNextRunDate = $oTask->Get('next_run_date');
|
|
||||||
$iNbRun = (int)$oTask->Get('total_exec_count');
|
|
||||||
$sAverageRunTime = $oTask->Get('average_run_duration');
|
|
||||||
$oP->p(sprintf('| %1$-25.25s | %2$-7s | %3$-19s | %4$-19s | %5$6d | %6$7s s |', $sTaskName, $sStatus,
|
|
||||||
$sLastRunDate, $sNextRunDate, $iNbRun, $sAverageRunTime));
|
|
||||||
}
|
|
||||||
$oP->p('+---------------------------+---------+---------------------+---------------------+--------+-----------+');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Arrange the list of processes in the best order for their execution.
|
|
||||||
* The idea is to continue just after the last task that was run, to let a chance to every task
|
|
||||||
* even when there are tasks taking a very long time (for example to process a big backlog)
|
|
||||||
* Note: We first record the last_run_date at the startup of a task, then at the end
|
|
||||||
* so that in case of a crash, the task is still listed has having run.
|
|
||||||
* In case the task crashes AND the previous task was very quick (less than 1 second)
|
|
||||||
* both tasks will have the same last_run_date. In this case it is important NOT to start again
|
|
||||||
* by the task that just crashed.
|
|
||||||
*
|
|
||||||
* @param iProcess[] $aProcesses
|
|
||||||
* @param BackgroundTask[] $aTasks
|
|
||||||
* @param DateTime $oNow
|
|
||||||
* @param $bVerbose
|
|
||||||
* @param Page $oP
|
|
||||||
*
|
|
||||||
* @throws \ArchivedObjectException
|
|
||||||
* @throws \CoreException
|
|
||||||
*/
|
|
||||||
function ReorderProcesses(&$aProcesses, $aTasks, $oNow, $bVerbose, &$oP)
|
|
||||||
{
|
|
||||||
$aIndexes = array_keys($aProcesses);
|
|
||||||
|
|
||||||
// Step 1: find which task was run last
|
|
||||||
$idx = 0;
|
|
||||||
$idxLastTaskExecuted = 0;
|
|
||||||
$sMaxRunDate = '';
|
|
||||||
if ($bVerbose)
|
|
||||||
{
|
|
||||||
$oP->p('Re-ordering the tasks - planned to run now - to continue after the last task run:');
|
|
||||||
$oP->p('+---------------------------+---------+---------------------+---------------------+');
|
|
||||||
$oP->p('| Task Class | Status | Last Run | Next Run |');
|
|
||||||
$oP->p('+---------------------------+---------+---------------------+---------------------+');
|
|
||||||
|
|
||||||
foreach($aProcesses as $sClass => $oProcess)
|
|
||||||
{
|
|
||||||
$sTaskClass = get_class($oProcess);
|
|
||||||
if (array_key_exists($sTaskClass, $aTasks))
|
|
||||||
{
|
|
||||||
$oTask = $aTasks[$sTaskClass];
|
$oTask = $aTasks[$sTaskClass];
|
||||||
if (($aTasks[$sTaskClass]->Get('status') == 'active') && ($aTasks[$sTaskClass]->Get('next_run_date') <= $oNow->format('Y-m-d H:i:s')))
|
if ($oTask->Get('next_run_date') == '3000-01-01 00:00:00')
|
||||||
{
|
{
|
||||||
$sTaskName = $oTask->Get('class_name');
|
// check for rescheduled tasks
|
||||||
$sStatus = $oTask->Get('status');
|
$oRefClass = new ReflectionClass($sTaskClass);
|
||||||
$sLastRunDate = $oTask->Get('latest_run_date');
|
if ($oRefClass->implementsInterface('iScheduledProcess'))
|
||||||
$sNextRunDate = $oTask->Get('next_run_date');
|
{
|
||||||
$oP->p(sprintf('| %1$-25.25s | %2$-7s | %3$-19s | %4$-19s |', $sTaskName, $sStatus, $sLastRunDate, $sNextRunDate));
|
$oNextOcc = $oProcess->GetNextOccurrence();
|
||||||
|
$oTask->Set('next_run_date', $oNextOcc->format('Y-m-d H:i:s'));
|
||||||
|
$oTask->DBUpdate();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
// Reactivate task if necessary
|
||||||
|
if ($oTask->Get('status') == 'removed')
|
||||||
|
{
|
||||||
|
$oTask->Set('status', 'active');
|
||||||
|
$oTask->DBUpdate();
|
||||||
|
}
|
||||||
|
// task having a real class to execute
|
||||||
|
unset($aTasks[$sTaskClass]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$oP->p('+---------------------------+---------+---------------------+---------------------+');
|
|
||||||
}
|
}
|
||||||
foreach($aProcesses as $sClass => $oProcess)
|
|
||||||
{
|
|
||||||
$sTaskClass = get_class($oProcess);
|
|
||||||
if (array_key_exists($sTaskClass, $aTasks))
|
|
||||||
{
|
|
||||||
$oTask = $aTasks[$sTaskClass];
|
|
||||||
|
|
||||||
if (($aTasks[$sTaskClass]->Get('status') == 'active') && ($aTasks[$sTaskClass]->Get('next_run_date') <= $oNow->format('Y-m-d H:i:s')))
|
// Remove all the tasks not having a valid class
|
||||||
{
|
foreach ($aTasks as $oTask)
|
||||||
if (($oTask->Get('latest_run_date') != '') && strcmp($oTask->Get('latest_run_date'), $sMaxRunDate) >= 0)
|
|
||||||
{
|
|
||||||
// More recent or equal (!important) will run later
|
|
||||||
$sMaxRunDate = $oTask->Get('latest_run_date');
|
|
||||||
$idxLastTaskExecuted = $idx;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$idx++;
|
|
||||||
}
|
|
||||||
if ($bVerbose)
|
|
||||||
{
|
|
||||||
$oLastRunProcess = $aProcesses[$aIndexes[$idxLastTaskExecuted]];
|
|
||||||
|
|
||||||
$oP->p('Last run process: '.get_class($oProcess)." (idx=$idxLastTaskExecuted) at ".$sMaxRunDate);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
$aReorderedProcesses = array();
|
|
||||||
|
|
||||||
|
|
||||||
// Step 2: the first task will the one just after the last run one, then the next, and so on (circular permutation)
|
|
||||||
$idx = 0;
|
|
||||||
$iTotal = count($aProcesses);
|
|
||||||
foreach($aProcesses as $oProcess)
|
|
||||||
{
|
{
|
||||||
$iActualIdx = (1 + $idxLastTaskExecuted + $idx )% $iTotal;
|
$sTaskClass = $oTask->Get('class_name');
|
||||||
$sKey = $aIndexes[$iActualIdx];
|
if (!class_exists($sTaskClass))
|
||||||
$aReorderedProcesses[] = $aProcesses[$sKey];
|
|
||||||
$idx++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($bVerbose)
|
|
||||||
{
|
|
||||||
$oP->p('After reordering, the execution order is:');
|
|
||||||
$oP->p('+---------------------------+---------+---------------------+---------------------+');
|
|
||||||
$oP->p('| Task Class | Status | Last Run | Next Run |');
|
|
||||||
$oP->p('+---------------------------+---------+---------------------+---------------------+');
|
|
||||||
|
|
||||||
foreach($aReorderedProcesses as $sClass => $oProcess)
|
|
||||||
{
|
{
|
||||||
$sTaskClass = get_class($oProcess);
|
$oTask->Set('status', 'removed');
|
||||||
if (array_key_exists($sTaskClass, $aTasks))
|
$oTask->DBUpdate();
|
||||||
{
|
|
||||||
$oTask = $aTasks[$sTaskClass];
|
|
||||||
if (($aTasks[$sTaskClass]->Get('status') == 'active') && ($aTasks[$sTaskClass]->Get('next_run_date') <= $oNow->format('Y-m-d H:i:s')))
|
|
||||||
{
|
|
||||||
$sTaskName = $oTask->Get('class_name');
|
|
||||||
$sStatus = $oTask->Get('status');
|
|
||||||
$sLastRunDate = $oTask->Get('latest_run_date');
|
|
||||||
$sNextRunDate = $oTask->Get('next_run_date');
|
|
||||||
$oP->p(sprintf('| %1$-25.25s | %2$-7s | %3$-19s | %4$-19s |', $sTaskName, $sStatus, $sLastRunDate, $sNextRunDate));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
$oP->p('+---------------------------+---------+---------------------+---------------------+');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update the array
|
if ($bVerbose)
|
||||||
$aProcesses = $aReorderedProcesses;
|
{
|
||||||
|
$aDisplayProcesses = array();
|
||||||
|
foreach ($aProcesses as $oExecInstance)
|
||||||
|
{
|
||||||
|
$aDisplayProcesses[] = get_class($oExecInstance);
|
||||||
|
}
|
||||||
|
$sDisplayProcesses = implode(', ', $aDisplayProcesses);
|
||||||
|
$oP->p("Background processes: ".$sDisplayProcesses);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
@@ -498,6 +482,51 @@ else
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
utils::UseParamFile();
|
utils::UseParamFile();
|
||||||
|
|
||||||
|
$bVerbose = utils::ReadParam('verbose', false, true /* Allow CLI */);
|
||||||
|
$bDebug = utils::ReadParam('debug', false, true /* Allow CLI */);
|
||||||
|
|
||||||
|
if ($bIsModeCLI)
|
||||||
|
{
|
||||||
|
// Next steps:
|
||||||
|
// specific arguments: 'csv file'
|
||||||
|
//
|
||||||
|
$sAuthUser = ReadMandatoryParam($oP, 'auth_user', 'raw_data');
|
||||||
|
$sAuthPwd = ReadMandatoryParam($oP, 'auth_pwd', 'raw_data');
|
||||||
|
if (UserRights::CheckCredentials($sAuthUser, $sAuthPwd))
|
||||||
|
{
|
||||||
|
UserRights::Login($sAuthUser); // Login & set the user's language
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$oP->p("Access wrong credentials ('$sAuthUser')");
|
||||||
|
$oP->output();
|
||||||
|
exit(EXIT_CODE_ERROR);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
require_once(APPROOT.'/application/loginwebpage.class.inc.php');
|
||||||
|
LoginWebPage::DoLogin(); // Check user rights and prompt if needed
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!UserRights::IsAdministrator())
|
||||||
|
{
|
||||||
|
$oP->p("Access restricted to administrators");
|
||||||
|
$oP->Output();
|
||||||
|
exit(EXIT_CODE_ERROR);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (utils::ReadParam('status_only', false, true /* Allow CLI */))
|
||||||
|
{
|
||||||
|
// Display status and exit
|
||||||
|
DisplayStatus($oP);
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
require_once(APPROOT.'core/mutex.class.inc.php');
|
||||||
|
$oP->p("Starting: ".time().' ('.date('Y-m-d H:i:s').')');
|
||||||
}
|
}
|
||||||
catch (Exception $e)
|
catch (Exception $e)
|
||||||
{
|
{
|
||||||
@@ -506,86 +535,10 @@ catch (Exception $e)
|
|||||||
exit(EXIT_CODE_FATAL);
|
exit(EXIT_CODE_FATAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($bIsModeCLI)
|
|
||||||
{
|
|
||||||
// Next steps:
|
|
||||||
// specific arguments: 'csvfile'
|
|
||||||
//
|
|
||||||
$sAuthUser = ReadMandatoryParam($oP, 'auth_user', 'raw_data');
|
|
||||||
$sAuthPwd = ReadMandatoryParam($oP, 'auth_pwd', 'raw_data');
|
|
||||||
if (UserRights::CheckCredentials($sAuthUser, $sAuthPwd))
|
|
||||||
{
|
|
||||||
UserRights::Login($sAuthUser); // Login & set the user's language
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
$oP->p("Access wrong credentials ('$sAuthUser')");
|
|
||||||
$oP->output();
|
|
||||||
exit(EXIT_CODE_ERROR);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
require_once(APPROOT.'/application/loginwebpage.class.inc.php');
|
|
||||||
LoginWebPage::DoLogin(); // Check user rights and prompt if needed
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!UserRights::IsAdministrator())
|
|
||||||
{
|
|
||||||
$oP->p("Access restricted to administrators");
|
|
||||||
$oP->Output();
|
|
||||||
exit(EXIT_CODE_ERROR);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Enumerate classes implementing BackgroundProcess
|
|
||||||
//
|
|
||||||
$aProcesses = array();
|
|
||||||
foreach (get_declared_classes() as $sPHPClass)
|
|
||||||
{
|
|
||||||
$oRefClass = new ReflectionClass($sPHPClass);
|
|
||||||
if ($oRefClass->isAbstract())
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
$oExtensionInstance = null;
|
|
||||||
if ($oRefClass->implementsInterface('iProcess'))
|
|
||||||
{
|
|
||||||
if (is_null($oExtensionInstance))
|
|
||||||
{
|
|
||||||
$oExecInstance = new $sPHPClass;
|
|
||||||
}
|
|
||||||
$aProcesses[$sPHPClass] = $oExecInstance;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
$bVerbose = utils::ReadParam('verbose', false, true /* Allow CLI */);
|
|
||||||
$bDebug = utils::ReadParam('debug', false, true /* Allow CLI */);
|
|
||||||
|
|
||||||
if ($bVerbose)
|
|
||||||
{
|
|
||||||
$aDisplayProcesses = array();
|
|
||||||
foreach ($aProcesses as $oExecInstance)
|
|
||||||
{
|
|
||||||
$aDisplayProcesses[] = get_class($oExecInstance);
|
|
||||||
}
|
|
||||||
$sDisplayProcesses = implode(', ', $aDisplayProcesses);
|
|
||||||
$oP->p("Background processes: ".$sDisplayProcesses);
|
|
||||||
}
|
|
||||||
if (utils::ReadParam('status_only', false, true /* Allow CLI */))
|
|
||||||
{
|
|
||||||
// Display status and exit
|
|
||||||
DisplayStatus($oP);
|
|
||||||
exit(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
require_once(APPROOT.'core/mutex.class.inc.php');
|
|
||||||
$oP->p("Starting: ".time().' ('.date('Y-m-d H:i:s').')');
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
$oConfig = utils::GetConfig();
|
|
||||||
$oMutex = new iTopMutex('cron');
|
$oMutex = new iTopMutex('cron');
|
||||||
|
$oConfig = utils::GetConfig();
|
||||||
if (!MetaModel::DBHasAccess(ACCESS_ADMIN_WRITE))
|
if (!MetaModel::DBHasAccess(ACCESS_ADMIN_WRITE))
|
||||||
{
|
{
|
||||||
$oP->p("A maintenance is ongoing");
|
$oP->p("A maintenance is ongoing");
|
||||||
@@ -594,7 +547,7 @@ try
|
|||||||
{
|
{
|
||||||
if ($oMutex->TryLock())
|
if ($oMutex->TryLock())
|
||||||
{
|
{
|
||||||
CronExec($oP, $aProcesses, $bVerbose);
|
CronExec($oP, $bVerbose);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -630,5 +583,4 @@ finally
|
|||||||
}
|
}
|
||||||
|
|
||||||
$oP->p("Exiting: ".time().' ('.date('Y-m-d H:i:s').')');
|
$oP->p("Exiting: ".time().' ('.date('Y-m-d H:i:s').')');
|
||||||
|
|
||||||
$oP->Output();
|
$oP->Output();
|
||||||
|
|||||||
Reference in New Issue
Block a user