ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
JobManagerImpl.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 
25 
26 readonly class JobManagerImpl implements \ILIAS\Cron\Job\JobManager
27 {
28  public function __construct(
29  private JobRepository $job_repository,
30  private \ilDBInterface $db,
31  private \ilSetting $settings,
32  private \ilLogger $logger,
33  private \ILIAS\Data\Clock\ClockFactory $clock_factory
34  ) {
35  }
36 
37  private function getMicrotime(): float
38  {
39  return ((int) $this->clock_factory->system()->now()->format('Uu')) / 1000000;
40  }
41 
42  public function runActiveJobs(\ilObjUser $actor): void
43  {
44  $this->logger->info('CRON - batch start');
45 
46  $ts = $this->clock_factory->system()->now()->getTimestamp();
47  $this->settings->set('last_cronjob_start_ts', (string) $ts);
48 
49  $useRelativeDates = \ilDatePresentation::useRelativeDates();
51  $this->logger->info(
52  \sprintf(
53  'Set last datetime to: %s',
55  )
56  );
57  $this->logger->info(
58  \sprintf(
59  'Verification of last run datetime (read from database): %s',
61  new \ilDateTime(\ilSetting::_lookupValue('common', 'last_cronjob_start_ts'), IL_CAL_UNIX)
62  )
63  )
64  );
66 
67  // ilLink::_getStaticLink() should work in crons
68  if (!\defined('ILIAS_HTTP_PATH')) {
69  \define('ILIAS_HTTP_PATH', \ilUtil::_getHttpPath());
70  }
71 
72  // system
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) {
76  // #18411 - we are NOT using the initial job data as it might be outdated at this point
77  $this->runJob($job, $actor);
78  }
79  }
80 
81  // plugins
82  foreach ($this->job_repository->getPluginJobs(true) as $item) {
83  // #18411 - we are NOT using the initial job data as it might be outdated at this point
84  $this->runJob($item[0], $actor);
85  }
86 
87  $this->logger->info('CRON - batch end');
88  }
89 
90  public function runJobManual(string $jobId, \ilObjUser $actor): bool
91  {
92  $result = false;
93 
94  $this->logger->info('CRON - manual start (' . $jobId . ')');
95 
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);
100  } else {
101  $this->logger->info('CRON - job ' . $jobId . ' is not intended to be executed manually');
102  }
103  } else {
104  $this->logger->info('CRON - job ' . $jobId . ' seems invalid or is inactive');
105  }
106 
107  $this->logger->info('CRON - manual end (' . $jobId . ')');
108 
109  return $result;
110  }
111 
117  private function runJob(
118  \ILIAS\Cron\CronJob $job,
119  \ilObjUser $actor,
120  ?array $jobData = null,
121  bool $isManualExecution = false
122  ): bool {
123  $did_run = false;
124 
125  if ($jobData === null) {
126  // aquire "fresh" job (status) data
127  $jobsData = $this->job_repository->getCronJobData($job->getId());
128  $jobData = array_pop($jobsData);
129  }
130 
131  $job->setDateTimeProvider(function (): \DateTimeImmutable {
132  return $this->clock_factory->system()->now();
133  });
134 
135  // already running?
136  if ($jobData['alive_ts']) {
137  $this->logger->info('CRON - job ' . $jobData['job_id'] . ' still running');
138 
139  $cut = 60 * 60 * 3;
140 
141  // is running (and has not pinged) for 3 hours straight, we assume it crashed
142  if ($this->clock_factory->system()->now()->getTimestamp() - ((int) $jobData['alive_ts']) > $cut) {
143  $this->job_repository->updateRunInformation($jobData['job_id'], 0, 0);
144  $this->deactivateJob($job, $actor); // #13082
145 
146  $result = new \ILIAS\Cron\Job\JobResult();
147  $result->setStatus(\ILIAS\Cron\Job\JobResult::STATUS_CRASHED);
148  $result->setCode(\ILIAS\Cron\Job\JobResult::CODE_SUPPOSED_CRASH);
149  $result->setMessage('Cron job deactivated because it has been inactive for 3 hours');
150 
151  $this->job_repository->updateJobResult(
152  $job,
153  $this->clock_factory->system()->now(),
154  $actor,
155  $result,
156  $isManualExecution
157  );
158 
159  $this->logger->info('CRON - job ' . $jobData['job_id'] . ' deactivated (assumed crash)');
160  }
161  } // initiate run?
162  elseif ($job->isDue(
163  $jobData['job_result_ts'] ? (new \DateTimeImmutable(
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']
168  ) : null,
169  $jobData['schedule_value'] ? (int) $jobData['schedule_value'] : null,
170  $isManualExecution
171  )) {
172  $this->logger->info('CRON - job ' . $jobData['job_id'] . ' started');
173 
174  $this->job_repository->updateRunInformation(
175  $jobData['job_id'],
176  $this->clock_factory->system()->now()->getTimestamp(),
177  $this->clock_factory->system()->now()->getTimestamp()
178  );
179 
180  $ts_in = $this->getMicrotime();
181  try {
182  $result = $job->run();
183  } catch (\Throwable $e) {
184  $result = new \ILIAS\Cron\Job\JobResult();
185  $result->setStatus(\ILIAS\Cron\Job\JobResult::STATUS_CRASHED);
186  $result->setMessage(
187  \ilStr::subStr(\sprintf('Exception: %s / %s', $e->getMessage(), $e->getTraceAsString()), 0, 400)
188  );
189 
190  $this->logger->error($e->getMessage());
191  $this->logger->error($e->getTraceAsString());
192  } finally {
193  $ts_dur = $this->getMicrotime() - $ts_in;
194  }
195 
196  if ($result->getStatus() === \ILIAS\Cron\Job\JobResult::STATUS_INVALID_CONFIGURATION) {
197  $this->deactivateJob($job, $actor);
198  $this->logger->info('CRON - job ' . $jobData['job_id'] . ' invalid configuration');
199  } else {
200  // success!
201  $did_run = true;
202  }
203 
204  $result->setDuration($ts_dur);
205 
206  $this->job_repository->updateJobResult(
207  $job,
208  $this->clock_factory->system()->now(),
209  $actor,
210  $result,
211  $isManualExecution
212  );
213  $this->job_repository->updateRunInformation($jobData['job_id'], 0, 0);
214 
215  $this->logger->info('CRON - job ' . $jobData['job_id'] . ' finished');
216  } else {
217  $this->logger->info('CRON - job ' . $jobData['job_id'] . ' returned status inactive');
218  }
219 
220  return $did_run;
221  }
222 
223  public function resetJob(\ILIAS\Cron\CronJob $job, \ilObjUser $actor): void
224  {
225  $result = new \ILIAS\Cron\Job\JobResult();
226  $result->setStatus(\ILIAS\Cron\Job\JobResult::STATUS_RESET);
227  $result->setCode(\ILIAS\Cron\Job\JobResult::CODE_MANUAL_RESET);
228  $result->setMessage('Cron job re-activated by admin');
229 
230  $this->job_repository->updateJobResult(
231  $job,
232  $this->clock_factory->system()->now(),
233  $actor,
234  $result,
235  true
236  );
237  $this->job_repository->resetJob($job);
238 
239  $this->activateJob($job, $actor, true);
240  }
241 
242  public function activateJob(\ILIAS\Cron\CronJob $job, \ilObjUser $actor, bool $wasManuallyExecuted = false): void
243  {
244  $this->job_repository->activateJob($job, $this->clock_factory->system()->now(), $actor, $wasManuallyExecuted);
245  $job->activationWasToggled($this->db, $this->settings, true);
246  }
247 
248  public function deactivateJob(\ILIAS\Cron\CronJob $job, \ilObjUser $actor, bool $wasManuallyExecuted = false): void
249  {
250  $this->job_repository->deactivateJob($job, $this->clock_factory->system()->now(), $actor, $wasManuallyExecuted);
251  $job->activationWasToggled($this->db, $this->settings, false);
252  }
253 
254  public function isJobActive(string $jobId): bool
255  {
256  $jobs_data = $this->job_repository->getCronJobData($jobId);
257 
258  return $jobs_data !== [] && $jobs_data[0]['job_status'];
259  }
260 
261  public function isJobInactive(string $jobId): bool
262  {
263  $jobs_data = $this->job_repository->getCronJobData($jobId);
264 
265  return $jobs_data !== [] && !((bool) $jobs_data[0]['job_status']);
266  }
267 
268  public function ping(string $jobId): void
269  {
270  $this->db->manipulateF(
271  'UPDATE cron_job SET alive_ts = %s WHERE job_id = %s',
272  ['integer', 'text'],
273  [$this->clock_factory->system()->now()->getTimestamp(), $jobId]
274  );
275  }
276 }
static array static setUseRelativeDates(bool $a_status)
set use relative dates
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)
Definition: class.ilStr.php:24
const IL_CAL_UNIX
static _lookupValue(string $a_module, string $a_keyword)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
__construct(private JobRepository $job_repository, private \ilDBInterface $db, private \ilSetting $settings, private \ilLogger $logger, private \ILIAS\Data\Clock\ClockFactory $clock_factory)
static _getHttpPath()
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)