19 declare(strict_types=1);
68 $this->
lng = $lng ?? $DIC->language();
69 $this->ilDB = $database ?? $DIC->database();
71 $this->
refinery = $refinery ?? $DIC->refinery();
72 $this->cronManager = $cronManager ?? $DIC->cron()->manager();
77 return 'frm_notification';
82 return $this->
lng->txt(
'cron_forum_notification');
87 return $this->
lng->txt(
'cron_forum_notification_crob_desc');
92 return JobScheduleType::IN_HOURS;
117 $this->
logger->debug(
'Sending ping to cron manager ...');
118 $this->cronManager->ping($this->
getId());
119 $this->
logger->debug(sprintf(
'Current memory usage: %s', memory_get_usage(
true)));
126 $this->
logger = $DIC->logger()->frm();
127 $this->tree = $DIC->repositoryTree();
129 $status = JobResult::STATUS_NO_ACTION;
131 $this->
lng->loadLanguageModule(
'forum');
133 $this->
logger->info(
'Started forum notification job ...');
135 if (!($last_run_datetime = $this->
settings->get(
'cron_forum_notification_last_date'))) {
136 $last_run_datetime =
null;
139 $this->num_sent_messages = 0;
140 $cj_start_date = date(
'Y-m-d H:i:s');
142 if ((
string) $this->
settings->get(
'max_notification_age',
'') ===
'') {
143 $this->
logger->info(sprintf(
144 'No maximum notification age set, %s days will be used to determine the ' .
145 'left interval when querying the relevant forum events.',
146 self::DEFAULT_MAX_NOTIFICATION_AGE_IN_DAYS
150 if ($last_run_datetime !==
null &&
152 (
int) date(
'm', strtotime($last_run_datetime)),
153 (
int) date(
'd', strtotime($last_run_datetime)),
154 (
int) date(
'Y', strtotime($last_run_datetime))
157 strtotime($last_run_datetime),
158 strtotime(
'-' . (
int) $this->
settings->get(
'max_notification_age', (
string) self::DEFAULT_MAX_NOTIFICATION_AGE_IN_DAYS) .
' days')
161 $threshold = strtotime(
'-' . (
int) $this->
settings->get(
'max_notification_age', (
string) self::DEFAULT_MAX_NOTIFICATION_AGE_IN_DAYS) .
' days');
164 $this->
logger->info(sprintf(
'Threshold for forum event determination is: %s', date(
'Y-m-d H:i:s', $threshold)));
166 $threshold_date = date(
'Y-m-d H:i:s', $threshold);
180 $this->
settings->set(
'cron_forum_notification_last_date', $cj_start_date);
182 $mess =
'Sent ' . $this->num_sent_messages .
' messages.';
184 $this->
logger->info($mess);
185 $this->
logger->info(
'Finished forum notification job');
188 if ($this->num_sent_messages !== 0) {
189 $status = JobResult::STATUS_OK;
190 $result->setMessage($mess);
193 $result->setStatus($status);
203 if (!array_key_exists($a_obj_id, self::$ref_ids_by_obj_id)) {
207 return self::$ref_ids_by_obj_id[$a_obj_id];
213 $ilAccess = $DIC->access();
215 if (!array_key_exists($a_user_id, self::$accessible_ref_ids_by_user)) {
216 self::$accessible_ref_ids_by_user[$a_user_id] = [];
219 if (!array_key_exists($a_obj_id, self::$accessible_ref_ids_by_user[$a_user_id])) {
220 $accessible_ref_id = 0;
222 if ($ilAccess->checkAccessOfUser($a_user_id,
'read',
'', $ref_id)) {
227 self::$accessible_ref_ids_by_user[$a_user_id][$a_obj_id] = $accessible_ref_id;
230 return (
int) self::$accessible_ref_ids_by_user[$a_user_id][$a_obj_id];
236 $ilDB = $DIC->database();
238 while ($row = $ilDB->fetchAssoc($res)) {
242 self::$deleted_ids_cache[$row[
'deleted_id']] = $row[
'deleted_id'];
253 'The recipient with id %s has no "read" permission for object with id %s',
264 $row[
'closest_container'] =
null;
269 $provider_id = isset($row[
'deleted_id']) ? -((
int) $row[
'deleted_id']) : (
int) $row[
'pos_pk'];
271 self::$providerObject[$provider_id .
'_' . $notification_type]->addRecipient((
int) $row[
'user_id']);
277 $usrIdsToPreload = [];
278 foreach (self::$providerObject as
$provider) {
279 if ($provider->getPosAuthorId() !== 0) {
280 $usrIdsToPreload[$provider->getPosAuthorId()] = $provider->getPosAuthorId();
282 if ($provider->getPosDisplayUserId() !== 0) {
283 $usrIdsToPreload[$provider->getPosDisplayUserId()] = $provider->getPosDisplayUserId();
285 if ($provider->getPostUpdateUserId() !== 0) {
286 $usrIdsToPreload[$provider->getPostUpdateUserId()] = $provider->getPostUpdateUserId();
293 foreach (self::$providerObject as $provider) {
294 if ($i > 0 && ($i % self::KEEP_ALIVE_CHUNK_SIZE) === 0) {
298 $recipients = array_unique($provider->getCronRecipients());
302 'Trying to send forum notifications for posting id "%s", type "%s" and recipients: %s',
303 $provider->getPostId(),
305 implode(
', ', $recipients)
310 $mailNotification->setIsCronjob(
true);
311 $mailNotification->setType($notification_type);
312 $mailNotification->setRecipients($recipients);
314 $mailNotification->send();
316 $this->num_sent_messages += count($provider->getCronRecipients());
317 $this->
logger->info(
'Sent notifications ... ');
328 public function determineClosestContainer(
int $frm_ref_id): ?
ilObject 330 if (isset(self::$container_by_frm_ref_id[$frm_ref_id])) {
331 return self::$container_by_frm_ref_id[$frm_ref_id];
334 $ref_id = $this->tree->checkForParentType($frm_ref_id,
'crs');
336 $ref_id = $this->tree->checkForParentType($frm_ref_id,
'grp');
342 self::$container_by_frm_ref_id[$frm_ref_id] =
$container;
351 return isset(self::$providerObject[$provider_id .
'_' . $notification_type]);
360 self::$providerObject[$provider_id .
'_' . $notification_type] = $tmp_provider;
361 self::$providerObject[$provider_id .
'_' . $notification_type]->addRecipient((
int) $row[
'user_id']);
366 self::$providerObject = [];
372 $a_fields[
'cron_forum_notification'] = $a_is_active ?
373 $this->
lng->txt(
'enabled') :
374 $this->
lng->txt(
'disabled');
382 if ($a_currently_active) {
386 $setting->
set(
'forum_notification', (
string) $value);
391 $this->
lng->loadLanguageModule(
'forum');
394 $this->
lng->txt(
'frm_max_notification_age'),
395 'max_notification_age' 397 $max_notification_age->
setSize(5);
398 $max_notification_age->setSuffix($this->
lng->txt(
'frm_max_notification_age_unit'));
399 $max_notification_age->setRequired(
true);
400 $max_notification_age->allowDecimals(
false);
401 $max_notification_age->setMinValue(1);
402 $max_notification_age->setInfo($this->
lng->txt(
'frm_max_notification_age_info'));
403 $max_notification_age->setValue(
405 'max_notification_age',
406 (
string) self::DEFAULT_MAX_NOTIFICATION_AGE_IN_DAYS
410 $a_form->
addItem($max_notification_age);
416 'max_notification_age',
419 $this->refinery->kindlyTo()->int(),
421 $this->
refinery->kindlyTo()->float(),
425 $this->
refinery->kindlyTo()->string()
435 frm_posts.pos_status = %s AND ( 436 (frm_posts.pos_date >= %s AND frm_posts.pos_date = frm_posts.pos_activation_date) OR 437 (frm_posts.pos_activation_date >= %s AND frm_posts.pos_date < frm_posts.pos_activation_date) 440 $values = [1, $threshold_date, $threshold_date];
442 $res = $this->ilDB->queryF(
458 frm_notification.interested_events & %s AND 459 frm_posts.pos_cens = %s AND frm_posts.pos_status = %s AND 460 (frm_posts.pos_update > frm_posts.pos_date AND frm_posts.pos_update >= %s) ';
469 $res = $this->ilDB->queryF(
485 frm_notification.interested_events & %s AND 486 frm_posts.pos_cens = %s AND frm_posts.pos_status = %s AND 487 (frm_posts.pos_cens_date >= %s AND frm_posts.pos_cens_date > frm_posts.pos_activation_date ) ';
496 $res = $this->ilDB->queryF(
512 frm_notification.interested_events & %s AND 513 frm_posts.pos_cens = %s AND frm_posts.pos_status = %s AND 514 (frm_posts.pos_cens_date >= %s AND frm_posts.pos_cens_date > frm_posts.pos_activation_date ) ';
523 $res = $this->ilDB->queryF(
531 'uncensored posting',
538 $res = $this->ilDB->queryF(
546 'frm_threads_deleted',
554 $res = $this->ilDB->queryF(
570 $numRows = $this->ilDB->numRows($res);
572 $this->
logger->info(sprintf(
'Sending notifications for %s "%s" events ...', $numRows, $actionName));
574 $this->
logger->info(sprintf(
'Sent notifications for %s ...', $actionName));
583 string $actionDescription,
584 int $notificationType
586 $numRows = $this->ilDB->numRows($res);
588 $this->
logger->info(sprintf(
'Sending notifications for %s "%s" events ...', $numRows, $actionDescription));
590 if (self::$deleted_ids_cache !== []) {
591 $this->ilDB->manipulate(
592 'DELETE FROM frm_posts_deleted WHERE ' . $this->ilDB->in(
594 self::$deleted_ids_cache,
599 $this->
logger->info(
'Deleted obsolete entries of table "' . $action .
'" ...');
601 $this->
logger->info(sprintf(
'Sent notifications for %s ...', $actionDescription));
610 SELECT frm_threads.thr_subject thr_subject, 611 frm_data.top_name top_name, 612 frm_data.top_frm_fk obj_id, 613 frm_notification.user_id user_id, 614 frm_threads.thr_pk thread_id, 616 FROM frm_notification, frm_posts, frm_threads, frm_data, frm_posts_tree 617 WHERE frm_posts.pos_thr_fk = frm_threads.thr_pk AND ' . $condition .
' 618 AND ((frm_threads.thr_top_fk = frm_data.top_pk AND frm_data.top_frm_fk = frm_notification.frm_id) 619 OR (frm_threads.thr_pk = frm_notification.thread_id 620 AND frm_data.top_pk = frm_threads.thr_top_fk) ) 621 AND frm_posts.pos_author_id != frm_notification.user_id 622 AND frm_posts_tree.pos_fk = frm_posts.pos_pk AND frm_posts_tree.parent_pos != 0 623 ORDER BY frm_posts.pos_date ASC';
629 SELECT frm_posts_deleted.thread_title thr_subject, 630 frm_posts_deleted.forum_title top_name, 631 frm_posts_deleted.obj_id obj_id, 632 frm_notification.user_id user_id, 633 frm_posts_deleted.pos_display_user_id, 634 frm_posts_deleted.pos_usr_alias, 635 frm_posts_deleted.deleted_id, 636 frm_posts_deleted.post_date pos_date, 637 frm_posts_deleted.post_title pos_subject, 638 frm_posts_deleted.post_message pos_message, 639 frm_posts_deleted.deleted_by 641 FROM frm_notification, frm_posts_deleted 643 WHERE ( frm_posts_deleted.obj_id = frm_notification.frm_id 644 OR frm_posts_deleted.thread_id = frm_notification.thread_id) 645 AND frm_posts_deleted.pos_display_user_id != frm_notification.user_id 646 AND frm_posts_deleted.is_thread_deleted = %s 647 AND frm_notification.interested_events & %s 648 ORDER BY frm_posts_deleted.post_date ASC';
readonly ilSetting $settings
existsProviderObject(int $provider_id, int $notification_type)
Interface Observer Contains several chained tasks and infos about them.
set(string $a_key, string $a_val)
static _getAllReferences(int $id)
get all reference ids for object ID
Class ilForumCronNotificationDataProvider.
sendNotificationForDeletedThreads()
static array $deleted_ids_cache
sendNotificationForUncensoredPosts(string $threshold_date)
sendDeleteNotifications(ilDBStatement $res, string $action, string $actionDescription, int $notificationType)
createSelectOfDeletionNotificationsSql()
const TYPE_THREAD_DELETED
const KEEP_ALIVE_CHUNK_SIZE
readonly ilDBInterface $ilDB
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
addProviderObject(int $provider_id, array $row, int $notification_type)
addCustomSettingsToForm(ilPropertyFormGUI $a_form)
const DEFAULT_MAX_NOTIFICATION_AGE_IN_DAYS
sendNotificationForCensoredPosts(string $threshold_date)
readonly JobManager $cronManager
static array $ref_ids_by_obj_id
getDefaultScheduleValue()
static getInstanceByRefId(int $ref_id, bool $stop_on_error=true)
get an instance of an Ilias object by reference id
activationWasToggled(ilDBInterface $db, ilSetting $setting, bool $a_currently_active)
Class ilForumNotificationCache.
static array $accessible_ref_ids_by_user
sendNotificationForNewPosts(string $threshold_date)
saveCustomSettings(ilPropertyFormGUI $a_form)
getRefIdsByObjId(int $a_obj_id)
readonly ILIAS Refinery Factory $refinery
readonly ilForumNotificationCache $notificationCache
__construct(?ilDBInterface $database=null, ?ilForumNotificationCache $notificationCache=null, ?ilLanguage $lng=null, ?ilSetting $settings=null, ?\ILIAS\Refinery\Factory $refinery=null, ?JobManager $cronManager=null)
addToExternalSettingsForm(int $a_form_id, array &$a_fields, bool $a_is_active)
static array $container_by_frm_ref_id
sendCronForumNotification(ilDBStatement $res, int $notification_type)
getFirstAccessibleRefIdBUserAndObjId(int $a_user_id, int $a_obj_id)
sendNotificationForUpdatedPosts(string $threshold_date)
sendNotification(ilDBStatement $res, string $actionName, int $notificationType)
const TYPE_POST_UNCENSORED
createForumPostSql(string $condition)
static array $providerObject
sendNotificationForDeletedPosts()