ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
JobTable.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
22 
27 
28 class JobTable implements \ILIAS\UI\Component\Table\DataRetrieval
29 {
30  private \ILIAS\UI\Factory $ui_factory;
31  private \ilLanguage $lng;
32  private \Psr\Http\Message\ServerRequestInterface $request;
33  private readonly \ILIAS\UI\URLBuilder $url_builder;
34  private readonly \ILIAS\UI\URLBuilderToken $action_parameter_token;
35  private readonly \ILIAS\UI\URLBuilderToken $row_id_token;
36 
40  public function __construct(
41  \ilCronManagerGUI $a_parent_obj,
42  string $a_parent_cmd,
43  array $table_action_namespace,
44  string $table_action_param_name,
45  string $table_row_identifier_name,
46  \ILIAS\UI\Factory $ui_factory,
47  \Psr\Http\Message\ServerRequestInterface $request,
48  \ilCtrlInterface $ctrl,
49  \ilLanguage $lng,
50  private readonly \ILIAS\Cron\Job\JobCollection $job_collection,
51  private readonly JobRepository $job_repository,
52  private readonly bool $mayWrite = false
53  ) {
54  $this->ui_factory = $ui_factory;
55  $this->lng = $lng;
56  $this->request = $request;
57 
58  $form_action = (new \ILIAS\Data\Factory())->uri(
59  \ilUtil::_getHttpPath() . '/' .
60  $ctrl->getLinkTarget($a_parent_obj, $a_parent_cmd)
61  );
62 
63  [
67  ] = (new \ILIAS\UI\URLBuilder($form_action))->acquireParameters(
68  $table_action_namespace,
69  $table_action_param_name,
70  $table_row_identifier_name
71  );
72  }
73 
74  public function getRows(
75  \ILIAS\UI\Component\Table\DataRowBuilder $row_builder,
76  array $visible_column_ids,
78  \ILIAS\Data\Order $order,
79  ?array $filter_data,
80  ?array $additional_parameters
81  ): \Generator {
82  foreach ($this->getRecords($range, $order) as $item) {
83  $record = [
84  'title' => $this->formatTitle($item),
85  'component' => $this->formatComponent($item),
86  'schedule' => $this->formatSchedule($item),
87  'status' => (bool) $item->getJobStatus(),
88  'status_info' => $this->formatStatusInfo($item),
89  'result' => $this->formatResult($item),
90  'result_info' => $this->formatResultInfo($item),
91  'last_run' => $this->formatLastRun($item)
92  ];
93 
94  if ($item->getJob()->hasFlexibleSchedule()) {
95  if ($item->getScheduleType() === null) {
96  $this->job_repository->updateJobSchedule(
97  $item->getJob(),
98  $item->getEffectiveScheduleType(),
99  $item->getEffectiveScheduleValue()
100  );
101  }
102  } elseif ($item->getScheduleType() !== null) {
103  $this->job_repository->updateJobSchedule($item->getJob(), null, null);
104  }
105 
106  $actions_executable = $this->mayWrite && !$item->getRunningTimestamp();
107  $is_crashed = \ILIAS\Cron\Job\JobResult::STATUS_CRASHED === $item->getJobResultStatus();
108  $is_acivated = (bool) $item->getJobStatus();
109 
110  $may_reset = $actions_executable && $is_crashed;
111  $may_activate = $actions_executable && !$is_crashed && !$is_acivated;
112  $may_deactivate = $actions_executable && !$is_crashed && $is_acivated;
113  $may_run = $actions_executable && !$is_crashed && $is_acivated && $item->getJob()->isManuallyExecutable();
114  $may_edit = $actions_executable && (
115  $item->getJob()->hasFlexibleSchedule() || $item->getJob()->hasCustomSettings()
116  );
117 
118  yield $row_builder
119  ->buildDataRow($item->getEffectiveJobId(), $record)
120  ->withDisabledAction('run', !$may_run)
121  ->withDisabledAction('activate', !$may_activate)
122  ->withDisabledAction('deactivate', !$may_deactivate)
123  ->withDisabledAction('reset', !$may_reset)
124  ->withDisabledAction('edit', !$may_edit);
125  }
126  }
127 
128  public function getTotalRowCount(?array $filter_data, ?array $additional_parameters): ?int
129  {
130  return \count($this->job_collection);
131  }
132 
136  private function getRecords(\ILIAS\Data\Range $range, \ILIAS\Data\Order $order): array
137  {
138  [$order_field, $order_direction] = $order->join([], static function ($ret, $key, $value) {
139  return [$key, $value];
140  });
141 
142  $collection = new OrderedJobEntities(
143  $this->job_collection,
144  match ($order_field) {
147  default => function (JobEntity $left, JobEntity $right) use ($order_field): int {
148  if ($order_field === 'component') {
149  return \ilStr::strCmp($this->formatComponent($left), $this->formatComponent($right));
150  }
151 
152  if ($order_field === 'schedule') {
153  return \ilStr::strCmp($this->formatSchedule($left), $this->formatSchedule($right));
154  }
155 
156  if ($order_field === 'result') {
157  return \ilStr::strCmp($this->formatResult($left), $this->formatResult($right));
158  }
159 
160  if ($order_field === 'last_run') {
161  $left_last_run = 0;
162  if ($left->getRunningTimestamp()) {
163  $left_last_run = strtotime('+1year', $left->getRunningTimestamp());
164  } elseif ($left->getJobResultTimestamp()) {
165  $left_last_run = $left->getJobResultTimestamp();
166  }
167 
168  $right_last_run = 0;
169  if ($right->getRunningTimestamp()) {
170  $right_last_run = strtotime('+1year', $right->getRunningTimestamp());
171  } elseif ($right->getJobResultTimestamp()) {
172  $right_last_run = $right->getJobResultTimestamp();
173  }
174 
175  return $left_last_run <=> $right_last_run;
176  }
177 
178  return 0;
179  }
180  },
181  $order_direction === \ILIAS\Data\Order::DESC
182  );
183 
184  $records = \array_slice($collection->toArray(), $range->getStart(), $range->getLength());
185 
186  return $records;
187  }
188 
189  private function formatTitle(JobEntity $entity): string
190  {
191  $title = implode('', [
192  '<span class="cron-title" id="job-' . $entity->getEffectiveJobId() . '">',
193  $entity->getEffectiveTitle(),
194  '</span>'
195  ]);
196  if ($entity->getJob()->getDescription()) {
197  $title .= implode('', [
198  '<div class="il_Description_no_margin">',
199  $entity->getJob()->getDescription(),
200  '</div>'
201  ]);
202  }
203 
204  return $title;
205  }
206 
207  private function formatSchedule(JobEntity $entity): string
208  {
209  $schedule = match ($entity->getEffectiveScheduleType()) {
210  JobScheduleType::DAILY => $this->lng->txt('cron_schedule_daily'),
211  JobScheduleType::WEEKLY => $this->lng->txt('cron_schedule_weekly'),
212  JobScheduleType::MONTHLY => $this->lng->txt('cron_schedule_monthly'),
213  JobScheduleType::QUARTERLY => $this->lng->txt('cron_schedule_quarterly'),
214  JobScheduleType::YEARLY => $this->lng->txt('cron_schedule_yearly'),
215  JobScheduleType::IN_MINUTES => \sprintf(
216  $this->lng->txt('cron_schedule_in_minutes'),
217  $entity->getEffectiveScheduleValue()
218  ),
219  JobScheduleType::IN_HOURS => \sprintf(
220  $this->lng->txt('cron_schedule_in_hours'),
221  $entity->getEffectiveScheduleValue()
222  ),
223  JobScheduleType::IN_DAYS => \sprintf(
224  $this->lng->txt('cron_schedule_in_days'),
225  $entity->getEffectiveScheduleValue()
226  )
227  };
228 
229  return $schedule;
230  }
231 
232  public function formatComponent(JobEntity $entity): string
233  {
234  $component = $entity->getComponent();
235  if ($entity->isPlugin()) {
236  $component = $this->lng->txt('cmps_plugin') . '/' . $component;
237  }
238 
239  return $component;
240  }
241 
242  private function formatStatusInfo(JobEntity $entity): string
243  {
244  $status_info = [];
245  if ($entity->getJobStatusTimestamp()) {
246  $status_info[] = \ilDatePresentation::formatDate(
248  );
249  }
250 
251  if ($entity->getJobStatusType()) {
252  $status_info[] = \ilUserUtil::getNamePresentation($entity->getJobStatusUsrId());
253  } else {
254  $status_info[] = $this->lng->txt('cron_changed_by_crontab');
255  }
256 
257  return implode('<br />', $status_info);
258  }
259 
260  private function formatResult(JobEntity $entity): string
261  {
262  $result = '-';
263  if ($entity->getJobResultStatus()) {
264  switch ($entity->getJobResultStatus()) {
265  case \ILIAS\Cron\Job\JobResult::STATUS_INVALID_CONFIGURATION:
266  $result = $this->lng->txt('cron_result_status_invalid_configuration');
267  break;
268 
269  case \ILIAS\Cron\Job\JobResult::STATUS_NO_ACTION:
270  $result = $this->lng->txt('cron_result_status_no_action');
271  break;
272 
273  case \ILIAS\Cron\Job\JobResult::STATUS_OK:
274  $result = $this->lng->txt('cron_result_status_ok');
275  break;
276 
277  case \ILIAS\Cron\Job\JobResult::STATUS_CRASHED:
278  $result = $this->lng->txt('cron_result_status_crashed');
279  break;
280 
281  case \ILIAS\Cron\Job\JobResult::STATUS_RESET:
282  $result = $this->lng->txt('cron_result_status_reset');
283  break;
284 
285  case \ILIAS\Cron\Job\JobResult::STATUS_FAIL:
286  $result = $this->lng->txt('cron_result_status_fail');
287  break;
288  }
289  }
290 
291  return $result;
292  }
293 
294  private function formatResultInfo(JobEntity $entity): string
295  {
296  $result_info = [];
297  if ($entity->getJobResultDuration()) {
298  $result_info[] = ($entity->getJobResultDuration() / 1000) . ' sec';
299  }
300 
301  // #23391 / #11866
302  $resultCode = $entity->getJobResultCode();
303  if (\in_array($resultCode, \ILIAS\Cron\Job\JobResult::getCoreCodes(), true)) {
304  $result_info[] = $this->lng->txt('cro_job_rc_' . $resultCode);
305  } elseif ($entity->getJobResultMessage()) {
306  $result_info[] = $entity->getJobResultMessage();
307  }
308 
309  if (\defined('DEVMODE') && DEVMODE && $resultCode) {
310  $result_info[] = $resultCode;
311  }
312 
313  if ($entity->getJobResultType()) {
314  $result_info[] = \ilUserUtil::getNamePresentation($entity->getJobResultUsrId());
315  } else {
316  $result_info[] = $this->lng->txt('cron_changed_by_crontab');
317  }
318 
319  return implode('<br />', $result_info);
320  }
321 
322  private function formatLastRun(JobEntity $entity): string
323  {
324  $last_run = null;
325  if ($entity->getRunningTimestamp()) {
326  $last_run = strtotime('+1year', $entity->getRunningTimestamp());
327  } elseif ($entity->getJobResultTimestamp()) {
328  $last_run = $entity->getJobResultTimestamp();
329  }
330 
331  if ($last_run > time()) {
332  $last_run = $this->lng->txt('cron_running_since') . ' ' .
334 
335  // job has pinged
336  if ($entity->getAliveTimestamp() !== $entity->getRunningTimestamp()) {
337  $last_run .= '<br />(Ping: ' .
339  }
340  } elseif ($last_run) {
341  $last_run = \ilDatePresentation::formatDate(new \ilDateTime($last_run, IL_CAL_UNIX));
342  }
343 
344  return $last_run ?: '-';
345  }
346 
350  public function getColumns(): array
351  {
352  return [
353  'title' => $this->ui_factory
354  ->table()
355  ->column()
356  ->text($this->lng->txt('title') . ' / ' . $this->lng->txt('description'))
357  ->withIsSortable(true),
358  'component' => $this->ui_factory
359  ->table()
360  ->column()
361  ->text($this->lng->txt('cron_component'))
362  ->withIsSortable(true)
363  ->withIsOptional(true, false),
364  'schedule' => $this->ui_factory
365  ->table()
366  ->column()
367  ->text($this->lng->txt('cron_schedule'))
368  ->withIsSortable(true),
369  'status' => $this->ui_factory
370  ->table()
371  ->column()
372  ->boolean(
373  $this->lng->txt('cron_status'),
374  $this->ui_factory->symbol()->icon()->custom(
375  'assets/images/standard/icon_ok.svg',
376  $this->lng->txt('cron_status_active'),
377  \ILIAS\UI\Component\Symbol\Icon\Icon::SMALL
378  ),
379  $this->ui_factory->symbol()->icon()->custom(
380  'assets/images/standard/icon_not_ok.svg',
381  $this->lng->txt('cron_status_inactive'),
382  \ILIAS\UI\Component\Symbol\Icon\Icon::SMALL
383  )
384  )
385  ->withIsSortable(true),
386  'status_info' => $this->ui_factory
387  ->table()
388  ->column()
389  ->text($this->lng->txt('cron_status_info'))
390  ->withIsSortable(false),
391  'result' => $this->ui_factory
392  ->table()
393  ->column()
394  ->status($this->lng->txt('cron_result'))
395  ->withIsSortable(true),
396  'result_info' => $this->ui_factory
397  ->table()
398  ->column()
399  ->text($this->lng->txt('cron_result_info'))
400  ->withIsSortable(false),
401  'last_run' => $this->ui_factory
402  ->table()
403  ->column()
404  ->text($this->lng->txt('cron_last_run'))
405  ->withIsSortable(true)
406  ];
407  }
408 
412  public function getActions(): array
413  {
414  if (!$this->mayWrite) {
415  return [];
416  }
417 
418  return [
419  'run' => $this->ui_factory->table()->action()->single(
420  $this->lng->txt('cron_action_run'),
421  $this->url_builder->withParameter($this->action_parameter_token, 'run'),
423  ),
424  'activate' => $this->ui_factory->table()->action()->standard(
425  $this->lng->txt('cron_action_activate'),
426  $this->url_builder->withParameter($this->action_parameter_token, 'activate'),
428  ),
429  'deactivate' => $this->ui_factory->table()->action()->standard(
430  $this->lng->txt('cron_action_deactivate'),
431  $this->url_builder->withParameter($this->action_parameter_token, 'deactivate'),
433  ),
434  'reset' => $this->ui_factory->table()->action()->standard(
435  $this->lng->txt('cron_action_reset'),
436  $this->url_builder->withParameter($this->action_parameter_token, 'reset'),
438  ),
439  'edit' => $this->ui_factory->table()->action()->single(
440  $this->lng->txt('cron_action_edit'),
441  $this->url_builder->withParameter($this->action_parameter_token, 'edit'),
443  )
444  ];
445  }
446 
447  public function getComponent(): \ILIAS\UI\Component\Table\Table
448  {
449  return $this->ui_factory
450  ->table()
451  ->data($this, $this->lng->txt('cron_jobs'), $this->getColumns())
452  ->withActions($this->getActions())
453  ->withId(self::class)
454  ->withRequest($this->request)
455  ->withOrder(new \ILIAS\Data\Order('title', \ILIAS\Data\Order::ASC));
456  }
457 }
readonly ILIAS UI URLBuilderToken $row_id_token
Definition: JobTable.php:35
Interface Observer Contains several chained tasks and infos about them.
getRecords(\ILIAS\Data\Range $range, \ILIAS\Data\Order $order)
Definition: JobTable.php:136
const IL_CAL_UNIX
getLinkTarget(object $a_gui_obj, ?string $a_cmd=null, ?string $a_anchor=null, bool $is_async=false, bool $has_xml_style=false)
Returns a link target for the given information.
Both the subject and the direction need to be specified when expressing an order. ...
Definition: Order.php:28
readonly ILIAS UI URLBuilderToken $action_parameter_token
Definition: JobTable.php:34
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
static getNamePresentation( $a_user_id, bool $a_user_image=false, bool $a_profile_link=false, string $a_profile_back_link='', bool $a_force_first_lastname=false, bool $a_omit_login=false, bool $a_sortable=true, bool $a_return_data_array=false, $a_ctrl_path='ilpublicuserprofilegui')
Default behaviour is:
Builds data types.
Definition: Factory.php:35
__construct(\ilCronManagerGUI $a_parent_obj, string $a_parent_cmd, array $table_action_namespace, string $table_action_param_name, string $table_row_identifier_name, \ILIAS\UI\Factory $ui_factory, \Psr\Http\Message\ServerRequestInterface $request, \ilCtrlInterface $ctrl, \ilLanguage $lng, private readonly \ILIAS\Cron\Job\JobCollection $job_collection, private readonly JobRepository $job_repository, private readonly bool $mayWrite=false)
Definition: JobTable.php:40
static _getHttpPath()
static formatDate(ilDateTime $date, bool $a_skip_day=false, bool $a_include_wd=false, bool $include_seconds=false, ?ilObjUser $user=null,)
getTotalRowCount(?array $filter_data, ?array $additional_parameters)
Mainly for the purpose of pagination-support, it is important to know about the total number of recor...
Definition: JobTable.php:128
ilCronManagerGUI: ilPropertyFormGUI ilCronManagerGUI: ilAdministrationGUI
const DESC
Definition: Order.php:31
A simple class to express a naive range of whole positive numbers.
Definition: Range.php:28
getRows(\ILIAS\UI\Component\Table\DataRowBuilder $row_builder, array $visible_column_ids, \ILIAS\Data\Range $range, \ILIAS\Data\Order $order, ?array $filter_data, ?array $additional_parameters)
Definition: JobTable.php:74
Psr Http Message ServerRequestInterface $request
Definition: JobTable.php:32