ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
CronJob.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
21 namespace ILIAS\Cron;
22 
25 
26 abstract class CronJob
27 {
29  protected ?int $schedule_value = null;
30  protected ?\Closure $date_time_provider = null;
31 
32  private function checkWeeklySchedule(\DateTimeImmutable $last_run, \DateTimeImmutable $now): bool
33  {
34  $last_year = (int) $last_run->format('Y');
35  $now_year = (int) $now->format('Y');
36 
37  if ($last_year > $now_year) {
38  // Should never happen, don't execute otherwise
39  return false;
40  }
41 
42  $last_week = $last_run->format('W');
43  $now_week = $now->format('W');
44 
45  if ($last_week !== $now_week) {
46  // Week differs, always execute the job
47  return true;
48  }
49 
50  // For all following cases, the week number is always identical
51 
52  $last_month = (int) $last_run->format('m');
53  $now_month = (int) $now->format('m');
54 
55  $is_within_same_week_in_same_year = ($last_year . '-' . $last_week) === ($now_year . '-' . $now_week);
56  if ($is_within_same_week_in_same_year) {
57  // Same week in same year, only execute if the month differs (2022-52 is valid for January and December)
58  return $last_month !== $now_month && $now->diff($last_run)->d > 7;
59  }
60 
61  if ($now_year - $last_year > 1) {
62  // Always execute if the difference of years is greater than 1
63  return true;
64  }
65 
66  // Execute for week number 52 in 2022 (last run) and week number 52 in December of 2022 (now), but not for week number 52 in January of 2022 (now)
67  return $last_month === $now_month;
68  }
69 
70  private function checkSchedule(
71  ?\DateTimeImmutable $last_run,
72  ?JobScheduleType $schedule_type,
73  ?int $schedule_value
74  ): bool {
75  if (null === $schedule_type) {
76  return false;
77  }
78 
79  if (null === $last_run) {
80  return true;
81  }
82 
83  if ($this->date_time_provider === null) {
84  $now = new \DateTimeImmutable('@' . time(), new \DateTimeZone(date_default_timezone_get()));
85  } else {
86  $now = ($this->date_time_provider)();
87  }
88 
89  switch ($schedule_type) {
90  case JobScheduleType::DAILY:
91  $last = $last_run->format('Y-m-d');
92  $ref = $now->format('Y-m-d');
93  return ($last !== $ref);
94 
95  case JobScheduleType::WEEKLY:
96  return $this->checkWeeklySchedule($last_run, $now);
97 
98  case JobScheduleType::MONTHLY:
99  $last = $last_run->format('Y-n');
100  $ref = $now->format('Y-n');
101  return ($last !== $ref);
102 
103  case JobScheduleType::QUARTERLY:
104  $last = $last_run->format('Y') . '-' . ceil(((int) $last_run->format('n')) / 3);
105  $ref = $now->format('Y') . '-' . ceil(((int) $now->format('n')) / 3);
106  return ($last !== $ref);
107 
109  $last = $last_run->format('Y');
110  $ref = $now->format('Y');
111  return ($last !== $ref);
112 
113  case JobScheduleType::IN_MINUTES:
114  $diff = floor(($now->getTimestamp() - $last_run->getTimestamp()) / 60);
115  return ($diff >= $schedule_value);
116 
117  case JobScheduleType::IN_HOURS:
118  $diff = floor(($now->getTimestamp() - $last_run->getTimestamp()) / (60 * 60));
119  return ($diff >= $schedule_value);
120 
121  case JobScheduleType::IN_DAYS:
122  $diff = floor(($now->getTimestamp() - $last_run->getTimestamp()) / (60 * 60 * 24));
123  return ($diff >= $schedule_value);
124  }
125 
126  return false;
127  }
128 
132  public function setDateTimeProvider(?\Closure $date_time_provider): void
133  {
134  if ($date_time_provider !== null) {
135  $r = new \ReflectionFunction($date_time_provider);
136  $return_type = $r->getReturnType();
137  if ($return_type instanceof \ReflectionNamedType) {
138  $return_type = $return_type->getName();
139  }
140  $expected_type = \DateTimeInterface::class;
141  if (!is_subclass_of($return_type, $expected_type)) {
142  throw new \InvalidArgumentException(
143  \sprintf(
144  'The return type of the datetime provider must be of type %s',
145  $expected_type
146  )
147  );
148  }
149 
150  $r = new \ReflectionFunction($date_time_provider);
151  $parameters = $r->getParameters();
152  if ($parameters !== []) {
153  throw new \InvalidArgumentException(
154  'The datetime provider must not define any parameters',
155  );
156  }
157  }
158 
159  $this->date_time_provider = $date_time_provider;
160  }
161 
162  public function isDue(
163  ?\DateTimeImmutable $last_run,
164  ?JobScheduleType $schedule_type,
165  ?int $schedule_value,
166  bool $is_manually_executed = false
167  ): bool {
168  if ($is_manually_executed) {
169  return true;
170  }
171 
172  if (!$this->hasFlexibleSchedule()) {
173  $schedule_type = $this->getDefaultScheduleType();
174  $schedule_value = $this->getDefaultScheduleValue();
175  }
176 
177  return $this->checkSchedule($last_run, $schedule_type, $schedule_value);
178  }
179 
183  public function getScheduleType(): ?JobScheduleType
184  {
185  if ($this->schedule_type && $this->hasFlexibleSchedule()) {
186  return $this->schedule_type;
187  }
188 
189  return null;
190  }
191 
195  public function getScheduleValue(): ?int
196  {
197  if ($this->schedule_value && $this->hasFlexibleSchedule()) {
198  return $this->schedule_value;
199  }
200 
201  return null;
202  }
203 
207  public function setSchedule(?JobScheduleType $a_type, ?int $a_value): void
208  {
209  if (
210  $a_value &&
211  $this->hasFlexibleSchedule() &&
212  \in_array($a_type, $this->getValidScheduleTypes(), true)
213  ) {
214  $this->schedule_type = $a_type;
215  $this->schedule_value = $a_value;
216  }
217  }
218 
223  public function getAllScheduleTypes(): array
224  {
225  return JobScheduleType::cases();
226  }
227 
231  public function getScheduleTypesWithValues(): array
232  {
233  return [
234  JobScheduleType::IN_MINUTES,
235  JobScheduleType::IN_HOURS,
236  JobScheduleType::IN_DAYS,
237  ];
238  }
239 
244  public function getValidScheduleTypes(): array
245  {
246  return $this->getAllScheduleTypes();
247  }
248 
249  public function isManuallyExecutable(): bool
250  {
251  return true;
252  }
253 
254  public function hasCustomSettings(): bool
255  {
256  return false;
257  }
258 
262  #[\Deprecated('Will be removed without any alternative, KS/UI forms will be expected', since: '13.0')]
263  public function usesLegacyForms(): bool
264  {
265  return true;
266  }
267 
269  \ILIAS\UI\Factory $ui_factory,
270  \ILIAS\Refinery\Factory $factory,
272  ): \ILIAS\UI\Component\Input\Container\Form\FormInput {
273  throw new \RuntimeException('Not implemented');
274  }
275 
279  #[\Deprecated('Will be removed without any alternative, KS/UI forms will be expected', since: '13.0')]
280  public function addCustomSettingsToForm(\ilPropertyFormGUI $a_form): void
281  {
282  }
283 
289  public function saveCustomConfiguration(mixed $form_data): void
290  {
291  throw new \RuntimeException('Not implemented');
292  }
293 
297  #[\Deprecated('Will be removed without any alternative, KS/UI forms will be expected', since: '13.0')]
298  public function saveCustomSettings(\ilPropertyFormGUI $a_form): bool
299  {
300  return true;
301  }
302 
306  public function addToExternalSettingsForm(int $a_form_id, array &$a_fields, bool $a_is_active): void
307  {
308  }
309 
314  public function activationWasToggled(\ilDBInterface $db, \ilSetting $setting, bool $a_currently_active): void
315  {
316  }
317 
318  abstract public function getId(): string;
319 
320  abstract public function getTitle(): string;
321 
322  abstract public function getDescription(): string;
323 
327  abstract public function hasAutoActivation(): bool;
328 
329  abstract public function hasFlexibleSchedule(): bool;
330 
331  abstract public function getDefaultScheduleType(): JobScheduleType;
332 
333  abstract public function getDefaultScheduleValue(): ?int;
334 
335  abstract public function run(): JobResult;
336 }
JobScheduleType $schedule_type
Definition: CronJob.php:28
activationWasToggled(\ilDBInterface $db, \ilSetting $setting, bool $a_currently_active)
Important: This method is (also) called from the setup process, where the constructor of an ilCronJob...
Definition: CronJob.php:314
getScheduleType()
Get current schedule type (if flexible)
Definition: CronJob.php:183
checkWeeklySchedule(\DateTimeImmutable $last_run, \DateTimeImmutable $now)
Definition: CronJob.php:32
Interface Observer Contains several chained tasks and infos about them.
isDue(?\DateTimeImmutable $last_run, ?JobScheduleType $schedule_type, ?int $schedule_value, bool $is_manually_executed=false)
Definition: CronJob.php:162
hasAutoActivation()
Is to be activated on "installation", does only work for ILIAS core cron jobs.
saveCustomConfiguration(mixed $form_data)
Definition: CronJob.php:289
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
ilSetting $setting
Definition: class.ilias.php:68
addToExternalSettingsForm(int $a_form_id, array &$a_fields, bool $a_is_active)
Definition: CronJob.php:306
getCustomConfigurationInput(\ILIAS\UI\Factory $ui_factory, \ILIAS\Refinery\Factory $factory, \ilLanguage $lng)
Definition: CronJob.php:268
setSchedule(?JobScheduleType $a_type, ?int $a_value)
Update current schedule (if flexible)
Definition: CronJob.php:207
addCustomSettingsToForm(\ilPropertyFormGUI $a_form)
Definition: CronJob.php:280
getScheduleValue()
Get current schedule value (if flexible)
Definition: CronJob.php:195
Builds data types.
Definition: Factory.php:35
saveCustomSettings(\ilPropertyFormGUI $a_form)
Definition: CronJob.php:298
getValidScheduleTypes()
Returns a collection of all valid schedule types for a specific job.
Definition: CronJob.php:244
setDateTimeProvider(?\Closure $date_time_provider)
Definition: CronJob.php:132
Closure $date_time_provider
Definition: CronJob.php:30
global $lng
Definition: privfeed.php:31
getScheduleTypesWithValues()
Definition: CronJob.php:231
getAllScheduleTypes()
Get all available schedule types.
Definition: CronJob.php:223
checkSchedule(?\DateTimeImmutable $last_run, ?JobScheduleType $schedule_type, ?int $schedule_value)
Definition: CronJob.php:70
$r