19 declare(strict_types=1);
39 return ((
int) $this->clock_factory->system()->now()->format(
'Uu')) / 1000000;
44 $this->
logger->info(
'CRON - batch start');
46 $ts = $this->clock_factory->system()->now()->getTimestamp();
47 $this->
settings->set(
'last_cronjob_start_ts', (
string) $ts);
53 'Set last datetime to: %s',
59 'Verification of last run datetime (read from database): %s',
68 if (!\defined(
'ILIAS_HTTP_PATH')) {
73 foreach ($this->job_repository->getCronJobData(
null,
false) as $row) {
74 $job = $this->job_repository->getJobInstanceById($row[
'job_id']);
75 if ($job instanceof \
ILIAS\
Cron\CronJob) {
77 $this->
runJob($job, $actor);
82 foreach ($this->job_repository->getPluginJobs(
true) as $item) {
84 $this->
runJob($item[0], $actor);
87 $this->
logger->info(
'CRON - batch end');
94 $this->
logger->info(
'CRON - manual start (' . $jobId .
')');
96 $job = $this->job_repository->getJobInstanceById($jobId);
97 if ($job instanceof \
ILIAS\
Cron\CronJob) {
98 if ($job->isManuallyExecutable()) {
99 $result = $this->
runJob($job, $actor,
null,
true);
101 $this->
logger->info(
'CRON - job ' . $jobId .
' is not intended to be executed manually');
104 $this->
logger->info(
'CRON - job ' . $jobId .
' seems invalid or is inactive');
107 $this->
logger->info(
'CRON - manual end (' . $jobId .
')');
120 ?array $jobData =
null,
121 bool $isManualExecution =
false 125 if ($jobData ===
null) {
127 $jobsData = $this->job_repository->getCronJobData($job->getId());
128 $jobData = array_pop($jobsData);
132 return $this->clock_factory->system()->now();
136 if ($jobData[
'alive_ts']) {
137 $this->
logger->info(
'CRON - job ' . $jobData[
'job_id'] .
' still running');
142 if ($this->clock_factory->system()->now()->getTimestamp() - ((
int) $jobData[
'alive_ts']) > $cut) {
143 $this->job_repository->updateRunInformation($jobData[
'job_id'], 0, 0);
146 $result = new \ILIAS\Cron\Job\JobResult();
149 $result->setMessage(
'Cron job deactivated because it has been inactive for 3 hours');
151 $this->job_repository->updateJobResult(
153 $this->clock_factory->system()->now(),
159 $this->
logger->info(
'CRON - job ' . $jobData[
'job_id'] .
' deactivated (assumed crash)');
164 '@' . $jobData[
'job_result_ts']
165 ))->setTimezone($this->clock_factory->system()->now()->getTimezone()) :
null,
166 is_numeric($jobData[
'schedule_type']) ? JobScheduleType::tryFrom(
167 (
int) $jobData[
'schedule_type']
169 $jobData[
'schedule_value'] ? (
int) $jobData[
'schedule_value'] :
null,
172 $this->
logger->info(
'CRON - job ' . $jobData[
'job_id'] .
' started');
174 $this->job_repository->updateRunInformation(
176 $this->clock_factory->system()->now()->getTimestamp(),
177 $this->clock_factory->system()->now()->getTimestamp()
182 $result = $job->run();
184 $result = new \ILIAS\Cron\Job\JobResult();
187 \
ilStr::subStr(\sprintf(
'Exception: %s / %s', $e->getMessage(), $e->getTraceAsString()), 0, 400)
190 $this->
logger->error($e->getMessage());
191 $this->
logger->error($e->getTraceAsString());
196 if ($result->getStatus() === \ILIAS\Cron\Job\JobResult::STATUS_INVALID_CONFIGURATION) {
198 $this->
logger->info(
'CRON - job ' . $jobData[
'job_id'] .
' invalid configuration');
204 $result->setDuration($ts_dur);
206 $this->job_repository->updateJobResult(
208 $this->clock_factory->system()->now(),
213 $this->job_repository->updateRunInformation($jobData[
'job_id'], 0, 0);
215 $this->
logger->info(
'CRON - job ' . $jobData[
'job_id'] .
' finished');
217 $this->
logger->info(
'CRON - job ' . $jobData[
'job_id'] .
' returned status inactive');
225 $result = new \ILIAS\Cron\Job\JobResult();
228 $result->setMessage(
'Cron job re-activated by admin');
230 $this->job_repository->updateJobResult(
232 $this->clock_factory->system()->now(),
237 $this->job_repository->resetJob($job);
244 $this->job_repository->activateJob($job, $this->clock_factory->system()->now(), $actor, $wasManuallyExecuted);
245 $job->activationWasToggled($this->db, $this->
settings,
true);
250 $this->job_repository->deactivateJob($job, $this->clock_factory->system()->now(), $actor, $wasManuallyExecuted);
251 $job->activationWasToggled($this->db, $this->
settings,
false);
256 $jobs_data = $this->job_repository->getCronJobData($jobId);
258 return $jobs_data !== [] && $jobs_data[0][
'job_status'];
263 $jobs_data = $this->job_repository->getCronJobData($jobId);
265 return $jobs_data !== [] && !((bool) $jobs_data[0][
'job_status']);
268 public function ping(
string $jobId): void
270 $this->db->manipulateF(
271 'UPDATE cron_job SET alive_ts = %s WHERE job_id = %s',
273 [$this->clock_factory->system()->now()->getTimestamp(), $jobId]
static array static setUseRelativeDates(bool $a_status)
set use relative dates
isJobInactive(string $jobId)
resetJob(\ILIAS\Cron\CronJob $job, \ilObjUser $actor)
runJob(\ILIAS\Cron\CronJob $job, \ilObjUser $actor, ?array $jobData=null, bool $isManualExecution=false)
Run single cron job (internal)
Interface Observer Contains several chained tasks and infos about them.
static subStr(string $a_str, int $a_start, ?int $a_length=null)
runActiveJobs(\ilObjUser $actor)
static useRelativeDates()
static _lookupValue(string $a_module, string $a_keyword)
final const string CODE_SUPPOSED_CRASH
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
isJobActive(string $jobId)
final const string CODE_MANUAL_RESET
final const int STATUS_CRASHED
__construct(private JobRepository $job_repository, private \ilDBInterface $db, private \ilSetting $settings, private \ilLogger $logger, private \ILIAS\Data\Clock\ClockFactory $clock_factory)
final const int STATUS_RESET
runJobManual(string $jobId, \ilObjUser $actor)
deactivateJob(\ILIAS\Cron\CronJob $job, \ilObjUser $actor, bool $wasManuallyExecuted=false)
static formatDate(ilDateTime $date, bool $a_skip_day=false, bool $a_include_wd=false, bool $include_seconds=false, ?ilObjUser $user=null,)
activateJob(\ILIAS\Cron\CronJob $job, \ilObjUser $actor, bool $wasManuallyExecuted=false)