ILIAS  release_8 Revision v8.19
All Data Structures Namespaces Files Functions Variables Modules Pages
class.ilChatroom.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
26 
34 {
35  public const ROOM_INVITATION = 'invitation_to_room';
36  private static string $settingsTable = 'chatroom_settings';
37  private static string $historyTable = 'chatroom_history';
38  private static string $userTable = 'chatroom_users';
39  private static string $sessionTable = 'chatroom_sessions';
40  private static string $banTable = 'chatroom_bans';
41  private static string $privateRoomsTable = 'chatroom_prooms';
42  private static string $privateSessionsTable = 'chatroom_psessions';
43  private static string $uploadTable = 'chatroom_uploads';
44  private static string $privateRoomsAccessTable = 'chatroom_proomaccess';
45  private array $settings = [];
52  private array $availableSettings = [
53  'object_id' => 'integer',
54  'online_status' => 'boolean',
55  'allow_anonymous' => 'boolean',
56  'allow_custom_usernames' => 'boolean',
57  'enable_history' => 'boolean',
58  'restrict_history' => 'boolean',
59  'autogen_usernames' => 'string',
60  'room_type' => 'string',
61  'allow_private_rooms' => 'boolean',
62  'display_past_msgs' => 'integer',
63  'private_rooms_enabled' => 'boolean'
64  ];
65  private int $roomId = 0;
66  private ?ilObjChatroom $object = null;
67 
75  public static function checkUserPermissions($permissions, int $ref_id, bool $send_info = true): bool
76  {
77  global $DIC;
78  $main_tpl = $DIC->ui()->mainTemplate();
79 
80  if (!is_array($permissions)) {
81  $permissions = [$permissions];
82  }
83 
84  $hasPermissions = self::checkPermissions($DIC->user()->getId(), $ref_id, $permissions);
85  if (!$hasPermissions && $send_info) {
86  $main_tpl->setOnScreenMessage('failure', $DIC->language()->txt('permission_denied'), true);
87 
88  return false;
89  }
90 
91  return $hasPermissions;
92  }
93 
102  public static function checkPermissionsOfUser(int $usr_id, $permissions, int $ref_id): bool
103  {
104  if (!is_array($permissions)) {
105  $permissions = [$permissions];
106  }
107 
108  return self::checkPermissions($usr_id, $ref_id, $permissions);
109  }
110 
117  protected static function checkPermissions(int $usrId, int $refId, array $permissions): bool
118  {
119  global $DIC;
120 
121  $pub_ref_id = ilObjChatroom::_getPublicRefId();
122 
123  foreach ($permissions as $permission) {
124  if ($pub_ref_id === $refId) {
125  $hasAccess = $DIC->rbac()->system()->checkAccessOfUser($usrId, $permission, $refId);
126  if ($hasAccess) {
127  $hasWritePermission = $DIC->rbac()->system()->checkAccessOfUser($usrId, 'write', $refId);
128  if ($hasWritePermission) {
129  continue;
130  }
131 
132  $visible = null;
133  $a_obj_id = ilObject::_lookupObjId($refId);
134  $active = ilObjChatroomAccess::isActivated($refId, $a_obj_id, $visible);
135 
136  switch ($permission) {
137  case 'visible':
138  if (!$active) {
139  $DIC->access()->addInfoItem(
141  $DIC->language()->txt('offline')
142  );
143  }
144 
145  if (!$active && !$visible) {
146  return false;
147  }
148  break;
149 
150  case 'read':
151  if (!$active) {
152  $DIC->access()->addInfoItem(
154  $DIC->language()->txt('offline')
155  );
156  return false;
157  }
158  break;
159  }
160  }
161  } else {
162  $hasAccess = $DIC->access()->checkAccessOfUser($usrId, $permission, '', $refId);
163  }
164 
165  if (!$hasAccess) {
166  return false;
167  }
168  }
169 
170  return true;
171  }
172 
173  public static function byObjectId(int $object_id): ?ilChatroom
174  {
175  global $DIC;
176 
177  $query = 'SELECT * FROM ' . self::$settingsTable . ' WHERE object_id = %s';
178  $types = ['integer'];
179  $values = [$object_id];
180  $rset = $DIC->database()->queryF($query, $types, $values);
181 
182  if ($row = $DIC->database()->fetchAssoc($rset)) {
183  $room = new self();
184  $room->initialize($row);
185  return $room;
186  }
187 
188  return null;
189  }
190 
196  public function initialize(array $rowdata): void
197  {
198  $this->roomId = (int) $rowdata['room_id'];
199 
200  foreach ($this->availableSettings as $setting => $type) {
201  if (isset($rowdata[$setting])) {
202  settype($rowdata[$setting], $this->availableSettings[$setting]);
203  $this->setSetting($setting, $rowdata[$setting]);
204  }
205  }
206  }
207 
213  public function setSetting(string $name, $value): void
214  {
215  $this->settings[$name] = $value;
216  }
217 
218  public static function byRoomId(int $room_id, bool $initObject = false): ?ilChatroom
219  {
220  global $DIC;
221 
222  $query = 'SELECT * FROM ' . self::$settingsTable . ' WHERE room_id = %s';
223 
224  $types = ['integer'];
225  $values = [$room_id];
226 
227  $rset = $DIC->database()->queryF($query, $types, $values);
228 
229  if ($row = $DIC->database()->fetchAssoc($rset)) {
230  $room = new self();
231  $room->initialize($row);
232 
233  if ($initObject) {
234  $room->object = ilObjectFactory::getInstanceByObjId((int) $row['object_id']);
235  }
236 
237  return $room;
238  }
239 
240  return null;
241  }
242 
246  public static function findDeletablePrivateRooms(): array
247  {
248  global $DIC;
249 
250  $query = '
251  SELECT private_rooms.proom_id id, MIN(disconnected) min_disconnected, MAX(disconnected) max_disconnected
252  FROM ' . self::$privateSessionsTable . ' private_sessions
253  INNER JOIN ' . self::$privateRoomsTable . ' private_rooms
254  ON private_sessions.proom_id = private_rooms.proom_id
255  WHERE closed = 0
256  GROUP BY private_rooms.proom_id
257  HAVING MIN(disconnected) > 0 AND MAX(disconnected) < %s';
258  $rset = $DIC->database()->queryF(
259  $query,
260  ['integer'],
261  [time() + 60 * 5]
262  );
263 
264  $rooms = [];
265 
266  while ($row = $DIC->database()->fetchAssoc($rset)) {
267  $rooms[$row['id']] = $row['id'];
268  }
269 
270  $query = 'SELECT DISTINCT proom_id, room_id, object_id FROM ' . self::$privateRoomsTable
271  . ' INNER JOIN ' . self::$settingsTable . ' ON parent_id = room_id '
272  . ' WHERE ' . $DIC->database()->in('proom_id', $rooms, false, 'integer');
273 
274  $rset = $DIC->database()->query($query);
275  $rooms = [];
276  while ($row = $DIC->database()->fetchAssoc($rset)) {
277  $rooms[] = [
278  'proom_id' => (int) $row['proom_id'],
279  'room_id' => (int) $row['room_id'],
280  'object_id' => (int) $row['object_id']
281  ];
282  }
283 
284  return $rooms;
285  }
286 
287  public function getDescription(): string
288  {
289  if (!$this->object) {
290  $this->object = ilObjectFactory::getInstanceByObjId((int) $this->getSetting('object_id'));
291  }
292 
293  return $this->object->getDescription();
294  }
295 
296  public function getSetting(string $name)
297  {
298  return $this->settings[$name];
299  }
300 
301  public function save(): void
302  {
303  $this->saveSettings($this->settings);
304  }
305 
306  public function saveSettings(array $settings): void
307  {
308  global $DIC;
309 
310  $localSettings = [];
311 
312  foreach ($this->availableSettings as $setting => $type) {
313  if (isset($settings[$setting])) {
314  if ($type === 'boolean') {
315  $settings[$setting] = (bool) $settings[$setting];
316  }
317  $localSettings[$setting] = [$this->phpTypeToMDBType($type), $settings[$setting]];
318  }
319  }
320 
321  if (!isset($localSettings['room_type']) || !$localSettings['room_type'][1]) {
322  $localSettings['room_type'][0] = 'text';
323  $localSettings['room_type'][1] = 'repository';
324  }
325 
326  if ($this->roomId > 0) {
327  $DIC->database()->update(
328  self::$settingsTable,
329  $localSettings,
330  ['room_id' => ['integer', $this->roomId]]
331  );
332  } else {
333  $this->roomId = $DIC->database()->nextId(self::$settingsTable);
334 
335  $localSettings['room_id'] = [
336  'integer', $this->roomId
337  ];
338 
339  $DIC->database()->insert(self::$settingsTable, $localSettings);
340  }
341  }
342 
343  private function phpTypeToMDBType(string $type): string
344  {
345  switch ($type) {
346  case 'string':
347  return 'text';
348 
349  case 'boolean':
350  return 'integer';
351 
352  default:
353  return $type;
354  }
355  }
356 
360  public function addHistoryEntry($message): void
361  {
362  global $DIC;
363 
364  $subRoom = 0;
365  $timestamp = 0;
366  if (is_array($message)) {
367  $subRoom = (int) ($message['sub'] ?? 0);
368  $timestamp = (int) $message['timestamp'];
369  } elseif (is_object($message)) {
370  $subRoom = (int) $message->sub;
371  $timestamp = (int) $message->timestamp;
372  }
373 
374  $id = $DIC->database()->nextId(self::$historyTable);
375  $DIC->database()->insert(
376  self::$historyTable,
377  [
378  'hist_id' => ['integer', $id],
379  'room_id' => ['integer', $this->roomId],
380  'sub_room' => ['integer', $subRoom],
381  'message' => ['text', json_encode($message, JSON_THROW_ON_ERROR)],
382  'timestamp' => ['integer', ($timestamp > 0 ? $timestamp : time())],
383  ]
384  );
385  }
386 
387  public function connectUser(ilChatroomUser $user): bool
388  {
389  global $DIC;
390 
391  $userdata = [
392  'login' => $user->getUsername(),
393  'id' => $user->getUserId()
394  ];
395 
396  $query = 'SELECT user_id FROM ' . self::$userTable . ' WHERE room_id = %s AND user_id = %s';
397  $types = ['integer', 'integer'];
398  $values = [$this->roomId, $user->getUserId()];
399 
400  if (!$DIC->database()->fetchAssoc($DIC->database()->queryF($query, $types, $values))) {
401  // Notice: Using replace instead of insert looks strange, because we actually know whether the selected data exists or not
402  // But we occasionally found some duplicate key errors although the data set should not exist when the following code is reached
403  $DIC->database()->replace(
404  self::$userTable,
405  [
406  'room_id' => ['integer', $this->roomId],
407  'user_id' => ['integer', $user->getUserId()]
408  ],
409  [
410  'userdata' => ['text', json_encode($userdata, JSON_THROW_ON_ERROR)],
411  'connected' => ['integer', time()],
412  ]
413  );
414 
415  return true;
416  }
417 
418  return false;
419  }
420 
421  public function getConnectedUsers(bool $only_data = true): array
422  {
423  global $DIC;
424 
425  $query = 'SELECT ' . ($only_data ? 'userdata' : '*') . ' FROM ' . self::$userTable . ' WHERE room_id = %s';
426  $types = ['integer'];
427  $values = [$this->roomId];
428  $rset = $DIC->database()->queryF($query, $types, $values);
429  $users = [];
430 
431  while ($row = $DIC->database()->fetchAssoc($rset)) {
432  $users[] = $only_data ? json_decode($row['userdata'], false, 512, JSON_THROW_ON_ERROR) : $row;
433  }
434 
435  return $users;
436  }
437 
438  public function disconnectUser(int $user_id): void
439  {
440  $this->disconnectUsers([$user_id]);
441  }
442 
446  public function disconnectUsers(array $userIds): void
447  {
448  global $DIC;
449 
450  $query = 'SELECT * FROM ' . self::$userTable . ' WHERE room_id = %s AND ' .
451  $DIC->database()->in('user_id', $userIds, false, 'integer');
452 
453  $types = ['integer'];
454  $values = [$this->roomId];
455  $res = $DIC->database()->queryF($query, $types, $values);
456 
457  if ($row = $DIC->database()->fetchAssoc($res)) {
458  $query = 'SELECT proom_id FROM ' . self::$privateRoomsTable . ' WHERE parent_id = %s';
459  $rset_prooms = $DIC->database()->queryF($query, ['integer'], [$this->roomId]);
460 
461  $prooms = [];
462 
463  while ($row_prooms = $DIC->database()->fetchAssoc($rset_prooms)) {
464  $prooms[] = $row_prooms['proom_id'];
465  }
466 
467  $query = 'UPDATE ' . self::$privateSessionsTable . ' SET disconnected = %s WHERE ' .
468  $DIC->database()->in('user_id', $userIds, false, 'integer') .
469  ' AND ' . $DIC->database()->in('proom_id', $prooms, false, 'integer');
470  $DIC->database()->manipulateF($query, ['integer'], [time()]);
471 
472  $query = 'DELETE FROM ' . self::$userTable . ' WHERE room_id = %s AND ' .
473  $DIC->database()->in('user_id', $userIds, false, 'integer');
474 
475  $types = ['integer'];
476  $values = [$this->roomId];
477  $DIC->database()->manipulateF($query, $types, $values);
478 
479  do {
480  if ($this->getSetting('enable_history')) {
481  $id = $DIC->database()->nextId(self::$sessionTable);
482  $DIC->database()->insert(
483  self::$sessionTable,
484  [
485  'sess_id' => ['integer', $id],
486  'room_id' => ['integer', $this->roomId],
487  'user_id' => ['integer', $row['user_id']],
488  'userdata' => ['text', $row['userdata']],
489  'connected' => ['integer', $row['connected']],
490  'disconnected' => ['integer', time()]
491  ]
492  );
493  }
494  } while ($row = $DIC->database()->fetchAssoc($res));
495  }
496  }
497 
498  public function getSettings(): array
499  {
500  return $this->settings;
501  }
502 
503  public function isSubscribed(int $chat_userid): bool
504  {
505  global $DIC;
506 
507  $query = 'SELECT COUNT(user_id) as cnt FROM ' . self::$userTable .
508  ' WHERE room_id = %s AND user_id = %s';
509 
510  $types = ['integer', 'integer'];
511  $values = [$this->roomId, $chat_userid];
512  $res = $DIC->database()->queryF($query, $types, $values);
513 
514  return ($row = $DIC->database()->fetchAssoc($res)) && (int) $row['cnt'] === 1;
515  }
516 
517  public function isAllowedToEnterPrivateRoom(int $chat_userid, int $proom_id): bool
518  {
519  global $DIC;
520 
521  $query = 'SELECT COUNT(user_id) cnt FROM ' . self::$privateRoomsAccessTable .
522  ' WHERE proom_id = %s AND user_id = %s';
523 
524  $types = ['integer', 'integer'];
525  $values = [$proom_id, $chat_userid];
526  $res = $DIC->database()->queryF($query, $types, $values);
527 
528  if (($row = $DIC->database()->fetchAssoc($res)) && (int) $row['cnt'] === 1) {
529  return true;
530  }
531 
532  $query = 'SELECT COUNT(*) cnt FROM ' . self::$privateRoomsTable .
533  ' WHERE proom_id = %s AND owner = %s';
534 
535  $types = ['integer', 'integer'];
536  $values = [$proom_id, $chat_userid];
537  $res = $DIC->database()->queryF($query, $types, $values);
538 
539  return ($row = $DIC->database()->fetchAssoc($res)) && (int) $row['cnt'] === 1;
540  }
541 
542  public function getHistory(
543  ilDateTime $from = null,
544  ilDateTime $to = null,
545  int $restricted_session_userid = null,
546  ?int $proom_id = 0,
547  bool $respect_target = true
548  ): array {
549  global $DIC;
550 
551  $join = '';
552 
553  if ($proom_id) {
554  $join .=
555  'INNER JOIN ' . self::$privateSessionsTable . ' pSessionTable ' .
556  'ON pSessionTable.user_id = ' . $DIC->database()->quote($restricted_session_userid, 'integer') . ' ' .
557  'AND pSessionTable.proom_id = historyTable.sub_room ' .
558  'AND timestamp >= pSessionTable.connected ' .
559  'AND timestamp <= pSessionTable.disconnected ';
560  }
561 
562  $query =
563  'SELECT historyTable.* ' .
564  'FROM ' . self::$historyTable . ' historyTable ' . $join . ' ' .
565  'WHERE historyTable.room_id = ' . $this->getRoomId();
566 
567  if ($proom_id !== null) {
568  $query .= ' AND historyTable.sub_room = ' . $DIC->database()->quote($proom_id, 'integer');
569  }
570 
571  $filter = [];
572 
573  if ($from !== null) {
574  $filter[] = 'timestamp >= ' . $DIC->database()->quote($from->getUnixTime(), 'integer');
575  }
576 
577  if ($to !== null) {
578  $filter[] = 'timestamp <= ' . $DIC->database()->quote($to->getUnixTime(), 'integer');
579  }
580 
581  if ($filter) {
582  $query .= ' AND ' . implode(' AND ', $filter);
583  }
584  $query .= ' ORDER BY timestamp ASC';
585 
586  $rset = $DIC->database()->query($query);
587  $result = [];
588 
589  while ($row = $DIC->database()->fetchAssoc($rset)) {
590  try {
591  $message = json_decode($row['message'], false, 512, JSON_THROW_ON_ERROR);
592  } catch (JsonException $e) {
593  $message = null;
594  } finally {
595  if ($message === null) {
596  $message = json_decode('{}', false, 512, JSON_THROW_ON_ERROR);
597  }
598  }
599 
600  $row['message'] = $message;
601  $row['message']->timestamp = $row['timestamp'];
602  if (
603  $respect_target &&
604  property_exists($row['message'], 'target') &&
605  $row['message']->target !== null &&
606  !$row['message']->target->public && (
607  !isset($row['recipients']) ||
608  !in_array($DIC->user()->getId(), explode(',', $row['recipients']), false)
609  )
610  ) {
611  continue;
612  }
613 
614  $result[] = $row;
615  }
616  return $result;
617  }
618 
619  public function getRoomId(): int
620  {
621  return $this->roomId;
622  }
623 
624  public function getPrivateRoomSessions(
625  ilDateTime $from,
626  ilDateTime $to,
627  int $user_id,
628  int $room_id
629  ): array {
630  global $DIC;
631 
632  $query = 'SELECT proom_id, title FROM ' . self::$privateRoomsTable . ' WHERE proom_id IN (
633  SELECT proom_id FROM ' . self::$privateSessionsTable . ' WHERE connected >= %s AND disconnected <= %s AND user_id = %s
634  ) AND parent_id = %s';
635 
636  $res = $DIC->database()->queryF(
637  $query,
638  ['integer', 'integer', 'integer', 'integer'],
639  [$from->getUnixTime(), $to->getUnixTime(), $user_id, $room_id]
640  );
641  $result = [];
642  while ($row = $DIC->database()->fetchAssoc($res)) {
643  $result[] = $row;
644  }
645 
646  return $result;
647  }
648 
649  public function saveFileUploadToDb(int $user_id, string $filename, string $type): void
650  {
651  global $DIC;
652 
653  $upload_id = $DIC->database()->nextId(self::$uploadTable);
654  $DIC->database()->insert(
655  self::$uploadTable,
656  [
657  'upload_id' => ['integer', $upload_id],
658  'room_id' => ['integer', $this->roomId],
659  'user_id' => ['integer', $user_id],
660  'filename' => ['text', $filename],
661  'filetype' => ['text', $type],
662  'timestamp' => ['integer', time()]
663  ]
664  );
665  }
666 
667  public function banUser(int $user_id, int $actor_id, string $comment = ''): void
668  {
669  global $DIC;
670 
671  $DIC->database()->replace(
672  self::$banTable,
673  [
674  'room_id' => ['integer', $this->roomId],
675  'user_id' => ['integer', $user_id]
676  ],
677  [
678  'actor_id' => ['integer', $actor_id],
679  'timestamp' => ['integer', time()],
680  'remark' => ['text', $comment]
681  ]
682  );
683  }
684 
690  public function unbanUser($user_id): int
691  {
692  global $DIC;
693 
694  if (!is_array($user_id)) {
695  $user_id = [$user_id];
696  }
697 
698  $query = 'DELETE FROM ' . self::$banTable . ' WHERE room_id = %s AND ' . $DIC->database()->in('user_id', $user_id, false, 'integer');
699  $types = ['integer'];
700  $values = [$this->getRoomId()];
701 
702  return $DIC->database()->manipulateF($query, $types, $values);
703  }
704 
705  public function isUserBanned(int $user_id): bool
706  {
707  global $DIC;
708 
709  $query = 'SELECT COUNT(user_id) cnt FROM ' . self::$banTable . ' WHERE user_id = %s AND room_id = %s';
710  $types = ['integer', 'integer'];
711  $values = [$user_id, $this->getRoomId()];
712 
713  $res = $DIC->database()->queryF($query, $types, $values);
714 
715  return ($row = $DIC->database()->fetchAssoc($res)) && $row['cnt'];
716  }
717 
718  public function getBannedUsers(): array
719  {
720  global $DIC;
721 
722  $query = 'SELECT chb.* FROM ' . self::$banTable . ' chb INNER JOIN usr_data ud ON chb.user_id = ud.usr_id WHERE chb.room_id = %s ';
723  $types = ['integer'];
724  $values = [$this->getRoomId()];
725  $res = $DIC->database()->queryF($query, $types, $values);
726  $result = [];
727 
728  while ($row = $DIC->database()->fetchAssoc($res)) {
729  if ($row['user_id'] > 0) {
730  $user = new ilObjUser((int) $row['user_id']);
731  $userdata = [
732  'user_id' => $user->getId(),
733  'firstname' => $user->getFirstname(),
734  'lastname' => $user->getLastname(),
735  'login' => $user->getLogin(),
736  'timestamp' => (int) $row['timestamp'],
737  'actor_id' => (int) $row['actor_id'],
738  'remark' => $row['remark']
739  ];
740 
741  $result[] = $userdata;
742  }
743  }
744 
745  return $result;
746  }
747 
748  public function getLastSession(ilChatroomUser $user): ?array
749  {
750  global $DIC;
751 
752  $query = 'SELECT * FROM ' . self::$sessionTable . ' WHERE user_id = ' .
753  $DIC->database()->quote($user->getUserId(), 'integer') .
754  ' ORDER BY connected DESC';
755 
756  $DIC->database()->setLimit(1);
757  $res = $DIC->database()->query($query);
758 
759  if ($row = $DIC->database()->fetchAssoc($res)) {
760  return $row;
761  }
762 
763  return null;
764  }
765 
766  public function getSessions(ilChatroomUser $user): array
767  {
768  global $DIC;
769 
770  $query = 'SELECT * FROM ' . self::$sessionTable
771  . ' WHERE room_id = ' .
772  $DIC->database()->quote($this->getRoomId(), 'integer') .
773  ' ORDER BY connected DESC';
774 
775  $res = $DIC->database()->query($query);
776 
777  $result = [];
778  while ($row = $DIC->database()->fetchAssoc($res)) {
779  $result[] = $row;
780  }
781 
782  return $result;
783  }
784 
785  public function addPrivateRoom(string $title, ilChatroomUser $owner, array $settings): int
786  {
787  global $DIC;
788 
789  $nextId = $DIC->database()->nextId(self::$privateRoomsTable);
790  $DIC->database()->insert(
791  self::$privateRoomsTable,
792  [
793  'proom_id' => ['integer', $nextId],
794  'parent_id' => ['integer', $this->roomId],
795  'title' => ['text', $title],
796  'owner' => ['integer', $owner->getUserId()],
797  'closed' => ['integer', ($settings['closed'] ?? 0)],
798  'created' => ['integer', ($settings['created'] ?? time())],
799  'is_public' => ['integer', $settings['public']],
800  ]
801  );
802 
803  return $nextId;
804  }
805 
806  public function closePrivateRoom(int $id): void
807  {
808  global $DIC;
809 
810  $DIC->database()->manipulateF(
811  'UPDATE ' . self::$privateRoomsTable . ' SET closed = %s WHERE proom_id = %s',
812  ['integer', 'integer'],
813  [time(), $id]
814  );
815  }
816 
817  public function isOwnerOfPrivateRoom(int $user_id, int $proom_id): bool
818  {
819  global $DIC;
820 
821  $query = 'SELECT proom_id FROM ' . self::$privateRoomsTable . ' WHERE proom_id = %s AND owner = %s';
822  $types = ['integer', 'integer'];
823  $values = [$proom_id, $user_id];
824 
825  $res = $DIC->database()->queryF($query, $types, $values);
826  if ($DIC->database()->fetchAssoc($res)) {
827  return true;
828  }
829 
830  return false;
831  }
832 
841  public function sendInvitationNotification(
842  ?ilChatroomObjectGUI $gui,
843  $sender,
844  int $recipient_id,
845  int $subScope = 0,
846  string $invitationLink = ''
847  ): void {
848  $links = [];
849 
850  if ($gui && $invitationLink === '') {
851  $invitationLink = $this->getChatURL($gui, $subScope);
852  }
853 
854  $links[] = new ilNotificationLink(
855  new ilNotificationParameter('chat_join', [], 'chatroom'),
856  $invitationLink
857  );
858 
859  if ($recipient_id > 0 && ANONYMOUS_USER_ID !== $recipient_id) {
860  if (is_numeric($sender) && $sender > 0) {
861  $sender_id = $sender;
863  $usr = ilObjectFactory::getInstanceByObjId($sender);
864  $public_name = $usr->getPublicName();
865  } elseif ($sender instanceof ilChatroomUser) {
866  if ($sender->getUserId() > 0) {
867  $sender_id = $sender->getUserId();
868  } else {
869  $sender_id = ANONYMOUS_USER_ID;
870  }
871  $public_name = $sender->getUsername();
872  } else {
873  throw new InvalidArgumentException(
874  '$sender must be an instance of ilChatroomUser or an id of an ilObjUser instance'
875  );
876  }
877 
878  $userLang = ilLanguageFactory::_getLanguageOfUser($recipient_id);
879  $userLang->loadLanguageModule('mail');
880  $bodyParams = [
881  'link' => $invitationLink,
882  'inviter_name' => $public_name,
883  'room_name' => $this->getTitle(),
884  'salutation' => ilMail::getSalutation($recipient_id, $userLang),
885  'BR' => "\n",
886  ];
887 
888  if ($subScope) {
889  $bodyParams['room_name'] .= ' - ' . self::lookupPrivateRoomTitle($subScope);
890  }
891 
892  $notification = new ilNotificationConfig(ChatInvitationNotificationProvider::NOTIFICATION_TYPE);
893  $notification->setTitleVar('chat_invitation', $bodyParams, 'chatroom');
894  $notification->setShortDescriptionVar('chat_invitation_short', $bodyParams, 'chatroom');
895  $notification->setLongDescriptionVar('chat_invitation_long', $bodyParams, 'chatroom');
896  $notification->setLinks($links);
897  $notification->setIconPath('templates/default/images/icon_chtr.svg');
898  $notification->setValidForSeconds(ilNotificationConfig::TTL_LONG);
899  $notification->setVisibleForSeconds(ilNotificationConfig::DEFAULT_TTS);
900  $notification->setIdentification(new NotificationIdentification(
901  ChatInvitationNotificationProvider::NOTIFICATION_TYPE,
902  self::ROOM_INVITATION . '_' . $this->getRefIdByRoomId($this->getRoomId()) . '_' . $subScope,
903  ));
904  $notification->setHandlerParam('mail.sender', (string) $sender_id);
905 
906  $notification->notifyByUsers([$recipient_id]);
907  }
908  }
909 
910  public function getChatURL(ilChatroomObjectGUI $gui, int $scope_id = 0): string
911  {
912  $url = '';
913  if ($scope_id) {
914  $url = ilLink::_getStaticLink($gui->getObject()->getRefId(), $gui->getObject()->getType(), true, '_' . $scope_id);
915  } else {
916  $url = ilLink::_getStaticLink($gui->getObject()->getRefId(), $gui->getObject()->getType());
917  }
918 
919  return $url;
920  }
921 
922  public function getTitle(): string
923  {
924  if (!$this->object) {
925  $this->object = ilObjectFactory::getInstanceByObjId((int) $this->getSetting('object_id'));
926  }
927 
928  return $this->object->getTitle();
929  }
930 
931  public static function lookupPrivateRoomTitle(int $proom_id): string
932  {
933  global $DIC;
934 
935  $query = 'SELECT title FROM ' . self::$privateRoomsTable . ' WHERE proom_id = %s';
936  $types = ['integer'];
937  $values = [$proom_id];
938 
939  $rset = $DIC->database()->queryF($query, $types, $values);
940  if ($row = $DIC->database()->fetchAssoc($rset)) {
941  return $row['title'];
942  }
943 
944  return 'unknown';
945  }
946 
947  public function inviteUserToPrivateRoomByLogin(string $login, int $proom_id): void
948  {
949  $user_id = (int) ilObjUser::_lookupId($login);
950  if ($user_id) {
951  $this->inviteUserToPrivateRoom($user_id, $proom_id);
952  }
953  }
954 
955  public function inviteUserToPrivateRoom(int $user_id, int $proom_id): void
956  {
957  global $DIC;
958 
959  $DIC->database()->replace(
960  self::$privateRoomsAccessTable,
961  [
962  'user_id' => ['integer', $user_id],
963  'proom_id' => ['integer', $proom_id]
964  ],
965  []
966  );
967  }
968 
969  public function getActivePrivateRooms(int $userid): array
970  {
971  global $DIC;
972 
973  $query = '
974  SELECT roomtable.title, roomtable.proom_id, accesstable.user_id id, roomtable.owner rowner
975  FROM ' . self::$privateRoomsTable . ' roomtable
976  LEFT JOIN ' . self::$privateRoomsAccessTable . ' accesstable
977  ON roomtable.proom_id = accesstable.proom_id
978  AND accesstable.user_id = %s
979  WHERE parent_id = %s
980  AND (closed = 0 OR closed IS NULL)
981  AND (accesstable.user_id IS NOT NULL OR roomtable.owner = %s)';
982  $types = ['integer', 'integer', 'integer'];
983  $values = [$userid, $this->roomId, $userid];
984  $rset = $DIC->database()->queryF($query, $types, $values);
985  $rooms = [];
986  while ($row = $DIC->database()->fetchAssoc($rset)) {
987  $row['active_users'] = $this->listUsersInPrivateRoom((int) $row['id']);
988  $row['owner'] = $row['rowner'];
989  $rooms[$row['proom_id']] = $row;
990  }
991 
992  return $rooms;
993  }
994 
999  public function listUsersInPrivateRoom(int $private_room_id): array
1000  {
1001  global $DIC;
1002 
1003  $query = '
1004  SELECT chatroom_users.user_id FROM ' . self::$privateSessionsTable . '
1005  INNER JOIN chatroom_users
1006  ON chatroom_users.user_id = ' . self::$privateSessionsTable . ' .user_id WHERE proom_id = %s AND disconnected = 0
1007  ';
1008  $types = ['integer'];
1009  $values = [$private_room_id];
1010  $rset = $DIC->database()->queryF($query, $types, $values);
1011 
1012  $users = [];
1013  while ($row = $DIC->database()->fetchAssoc($rset)) {
1014  $users[(int) $row['user_id']] = (int) $row['user_id'];
1015  }
1016 
1017  return array_values($users);
1018  }
1019 
1020  public function subscribeUserToPrivateRoom(int $room_id, int $user_id): void
1021  {
1022  global $DIC;
1023 
1024  if (!$this->userIsInPrivateRoom($room_id, $user_id)) {
1025  $id = $DIC->database()->nextId(self::$privateSessionsTable);
1026  $DIC->database()->insert(
1027  self::$privateSessionsTable,
1028  [
1029  'psess_id' => ['integer', $id],
1030  'proom_id' => ['integer', $room_id],
1031  'user_id' => ['integer', $user_id],
1032  'connected' => ['integer', time()],
1033  'disconnected' => ['integer', 0],
1034  ]
1035  );
1036  }
1037  }
1038 
1039  public function userIsInPrivateRoom(int $room_id, int $user_id): bool
1040  {
1041  global $DIC;
1042 
1043  $query = 'SELECT proom_id id FROM ' . self::$privateSessionsTable .
1044  ' WHERE user_id = %s AND proom_id = %s AND disconnected = 0';
1045  $types = ['integer', 'integer'];
1046  $values = [$user_id, $room_id];
1047  $rset = $DIC->database()->queryF($query, $types, $values);
1048  if ($DIC->database()->fetchAssoc($rset)) {
1049  return true;
1050  }
1051 
1052  return false;
1053  }
1054 
1055  public function unsubscribeUserFromPrivateRoom(int $room_id, int $user_id): void
1056  {
1057  global $DIC;
1058 
1059  $DIC->database()->update(
1060  self::$privateSessionsTable,
1061  [
1062  'disconnected' => ['integer', time()]
1063  ],
1064  [
1065  'proom_id' => ['integer', $room_id],
1066  'user_id' => ['integer', $user_id]
1067  ]
1068  );
1069  }
1070 
1071  public function countActiveUsers(): int
1072  {
1073  global $DIC;
1074 
1075  $query = 'SELECT COUNT(user_id) cnt FROM ' . self::$userTable . ' WHERE room_id = %s';
1076  $types = ['integer'];
1077  $values = [$this->roomId];
1078  $res = $DIC->database()->queryF($query, $types, $values);
1079 
1080  if ($row = $DIC->database()->fetchAssoc($res)) {
1081  return (int) $row['cnt'];
1082  }
1083 
1084  return 0;
1085  }
1086 
1087  public function getPrivateRooms(): array
1088  {
1089  global $DIC;
1090 
1091  $query = 'SELECT * FROM ' . self::$privateRoomsTable . ' WHERE parent_id = %s';
1092  $rset = $DIC->database()->queryF($query, ['integer'], [$this->roomId]);
1093 
1094  $rooms = [];
1095  while ($row = $DIC->database()->fetchAssoc($rset)) {
1096  $rooms[] = $row;
1097  }
1098 
1099  return $rooms;
1100  }
1101 
1106  public function getPrivilegedUsersForPrivateRoom(int $subRoomId): array
1107  {
1108  global $DIC;
1109 
1110  $query = 'SELECT user_id FROM ' . self::$privateRoomsAccessTable . ' WHERE proom_id = %s';
1111  $rset = $DIC->database()->queryF($query, ['integer'], [$subRoomId]);
1112 
1113  $userIds = [];
1114  while ($row = $DIC->database()->fetchAssoc($rset)) {
1115  $userIds[] = (int) $row['user_id'];
1116  }
1117 
1118  return $userIds;
1119  }
1120 
1121  public function getUniquePrivateRoomTitle(string $title): string
1122  {
1123  global $DIC;
1124 
1125  $query = 'SELECT title FROM ' . self::$privateRoomsTable . ' WHERE parent_id = %s and closed = 0';
1126  $rset = $DIC->database()->queryF($query, ['integer'], [$this->roomId]);
1127 
1128  $titles = [];
1129  while ($row = $DIC->database()->fetchAssoc($rset)) {
1130  $titles[] = $row['title'];
1131  }
1132 
1133  $suffix = '';
1134  $i = 0;
1135  do {
1136  if (!in_array($title . $suffix, $titles, true)) {
1137  $title .= $suffix;
1138  break;
1139  }
1140 
1141  ++$i;
1142 
1143  $suffix = ' (' . $i . ')';
1144  } while (true);
1145 
1146  return $title;
1147  }
1148 
1154  public function getAccessibleRoomIdByTitleMap(int $user_id): array
1155  {
1156  global $DIC;
1157 
1158  $query = "
1159  SELECT room_id, od.title, objr.ref_id
1160  FROM object_data od
1161  INNER JOIN " . self::$settingsTable . "
1162  ON object_id = od.obj_id
1163  INNER JOIN object_reference objr
1164  ON objr.obj_id = od.obj_id
1165  AND objr.deleted IS NULL
1166  INNER JOIN tree
1167  ON tree.child = objr.ref_id
1168  AND tree.tree = %s
1169  WHERE od.type = %s
1170  ";
1171 
1172  $types = ['integer', 'text'];
1173  $values = [1, 'chtr'];
1174  $res = $DIC->database()->queryF($query, $types, $values);
1175 
1176  $rooms = [];
1177  while ($row = $DIC->database()->fetchAssoc($res)) {
1178  if (self::checkPermissionsOfUser($user_id, 'read', (int) $row['ref_id'])) {
1179  $rooms[(int) $row['room_id']] = $row['title'];
1180  }
1181  }
1182 
1183  return $rooms;
1184  }
1185 
1191  public function getPrivateSubRooms(int $parent_room, int $user_id): array
1192  {
1193  global $DIC;
1194 
1195  $query = "
1196  SELECT proom_id, parent_id
1197  FROM " . self::$privateRoomsTable . "
1198  WHERE parent_id = %s
1199  AND owner = %s
1200  AND closed = 0
1201  ";
1202 
1203  $types = ['integer', 'integer'];
1204  $values = [$parent_room, $user_id];
1205  $res = $DIC->database()->queryF($query, $types, $values);
1206 
1207  $priv_rooms = [];
1208  while ($row = $DIC->database()->fetchAssoc($res)) {
1209  $proom_id = (int) $row['proom_id'];
1210  $priv_rooms[$proom_id] = (int) $row['parent_id'];
1211  }
1212 
1213  return $priv_rooms;
1214  }
1215 
1216  public function getRefIdByRoomId(int $room_id): int
1217  {
1218  global $DIC;
1219 
1220  $query = "
1221  SELECT objr.ref_id
1222  FROM object_reference objr
1223 
1224  INNER JOIN chatroom_settings cs
1225  ON cs.object_id = objr.obj_id
1226 
1227  INNER JOIN object_data od
1228  ON od.obj_id = cs.object_id
1229 
1230  WHERE cs.room_id = %s
1231  ";
1232 
1233  $types = ['integer'];
1234  $values = [$room_id];
1235 
1236  $res = $DIC->database()->queryF($query, $types, $values);
1237 
1238  $row = $DIC->database()->fetchAssoc($res);
1239 
1240  return (int) ($row['ref_id'] ?? 0);
1241  }
1242 
1243  public function getLastMessages(int $number, ilChatroomUser $chatuser): array
1244  {
1245  global $DIC;
1246 
1247  // There is currently no way to check if a message is private or not
1248  // by sql. So we fetch twice as much as we need and hope that there
1249  // are not more than $number private messages.
1250  $DIC->database()->setLimit($number);
1251  $rset = $DIC->database()->query(
1252  'SELECT *
1253  FROM ' . self::$historyTable . '
1254  WHERE room_id = ' . $DIC->database()->quote($this->roomId, 'integer') . '
1255  AND sub_room = 0
1256  AND (
1257  (' . $DIC->database()->like('message', 'text', '%"type":"message"%') . ' AND NOT ' . $DIC->database()->like('message', 'text', '%"public":0%') . ')
1258  OR ' . $DIC->database()->like('message', 'text', '%"target":{%"id":"' . $chatuser->getUserId() . '"%') . '
1259  OR ' . $DIC->database()->like('message', 'text', '%"from":{"id":' . $chatuser->getUserId() . '%') . '
1260  )
1261  ORDER BY timestamp DESC'
1262  );
1263 
1264  $result_count = 0;
1265  $results = [];
1266  while (($row = $DIC->database()->fetchAssoc($rset)) && $result_count < $number) {
1267  $tmp = json_decode($row['message'], false, 512, JSON_THROW_ON_ERROR);
1268  if (property_exists($tmp, 'target') && $tmp->target instanceof stdClass && (int) $tmp->target->public === 0) {
1269  if (in_array($chatuser->getUserId(), [(int) $tmp->target->id, (int) $tmp->from->id], true)) {
1270  $results[] = $tmp;
1271  ++$result_count;
1272  }
1273  } else {
1274  $results[] = $tmp;
1275  ++$result_count;
1276  }
1277  }
1278 
1279  if ($results !== []) {
1280  $rset = $DIC->database()->queryF(
1281  'SELECT *
1282  FROM ' . self::$historyTable . '
1283  WHERE room_id = %s
1284  AND sub_room = 0
1285  AND ' . $DIC->database()->like('message', 'text', '%%"type":"notice"%%') . '
1286  AND timestamp <= %s AND timestamp >= %s
1287  ORDER BY timestamp DESC',
1288  ['integer', 'integer', 'integer'],
1289  [$this->roomId, $results[0]->timestamp, $results[$result_count - 1]->timestamp]
1290  );
1291 
1292  while (($row = $DIC->database()->fetchAssoc($rset))) {
1293  $tmp = json_decode($row['message'], false, 512, JSON_THROW_ON_ERROR);
1294  $results[] = $tmp;
1295  }
1296  }
1297 
1298  usort($results, static function (stdClass $a, stdClass $b): int {
1299  $a_timestamp = strlen((string) $a->timestamp) === 13 ? ((int) substr((string) $a->timestamp, 0, -3)) : $a->timestamp;
1300  $b_timestamp = strlen((string) $b->timestamp) === 13 ? ((int) substr((string) $b->timestamp, 0, -3)) : $b->timestamp;
1301 
1302  return $b_timestamp - $a_timestamp;
1303  });
1304 
1305  return $results;
1306  }
1307 
1308  public function clearMessages(int $sub_room): void
1309  {
1310  global $DIC;
1311 
1312  $DIC->database()->queryF(
1313  'DELETE FROM ' . self::$historyTable . ' WHERE room_id = %s AND sub_room = %s',
1314  ['integer', 'integer'],
1315  [$this->roomId, $sub_room]
1316  );
1317 
1318  if ($sub_room) {
1319  $DIC->database()->queryF(
1320  'DELETE FROM ' . self::$privateSessionsTable . ' WHERE proom_id = %s AND disconnected < %s',
1321  ['integer', 'integer'],
1322  [$sub_room, time()]
1323  );
1324  } else {
1325  $DIC->database()->queryF(
1326  'DELETE FROM ' . self::$sessionTable . ' WHERE room_id = %s AND disconnected < %s',
1327  ['integer', 'integer'],
1328  [$this->roomId, time()]
1329  );
1330  }
1331  }
1332 }
static checkUserPermissions($permissions, int $ref_id, bool $send_info=true)
Checks user permissions by given array and ref_id.
static string $privateSessionsTable
getSetting(string $name)
$res
Definition: ltiservices.php:69
getPrivateRoomSessions(ilDateTime $from, ilDateTime $to, int $user_id, int $room_id)
getPrivilegedUsersForPrivateRoom(int $subRoomId)
getUserId()
Returns Ilias User ID.
static string $banTable
const ROOM_INVITATION
const ANONYMOUS_USER_ID
Definition: constants.php:27
$type
description of a localized parameter this information is used locate translations while processing no...
getRefIdByRoomId(int $room_id)
banUser(int $user_id, int $actor_id, string $comment='')
getLastSession(ilChatroomUser $user)
static lookupPrivateRoomTitle(int $proom_id)
getLastMessages(int $number, ilChatroomUser $chatuser)
array $availableSettings
disconnectUsers(array $userIds)
getPrivateSubRooms(int $parent_room, int $user_id)
static _lookupId($a_user_str)
$refId
Definition: xapitoken.php:58
static string $historyTable
getConnectedUsers(bool $only_data=true)
inviteUserToPrivateRoomByLogin(string $login, int $proom_id)
phpTypeToMDBType(string $type)
setSetting(string $name, $value)
Sets given name and value as setting into $this->settings array.
ilObjChatroom $object
static _lookupObjId(int $ref_id)
global $DIC
Definition: feed.php:28
saveFileUploadToDb(int $user_id, string $filename, string $type)
if($format !==null) $name
Definition: metadata.php:247
getHistory(ilDateTime $from=null, ilDateTime $to=null, int $restricted_session_userid=null, ?int $proom_id=0, bool $respect_target=true)
$ref_id
Definition: ltiauth.php:67
static string $settingsTable
clearMessages(int $sub_room)
addHistoryEntry($message)
static checkPermissionsOfUser(int $usr_id, $permissions, int $ref_id)
Checks user permissions in question for a given user id in relation to a given ref_id.
static _getLanguageOfUser(int $a_usr_id)
Get language object of user.
static isActivated(int $refId, int $objId, bool &$a_visible_flag=null)
isSubscribed(int $chat_userid)
static getSalutation(int $a_usr_id, ?ilLanguage $a_language=null)
getSessions(ilChatroomUser $user)
static findDeletablePrivateRooms()
addPrivateRoom(string $title, ilChatroomUser $owner, array $settings)
isUserBanned(int $user_id)
static string $privateRoomsTable
listUsersInPrivateRoom(int $private_room_id)
static byRoomId(int $room_id, bool $initObject=false)
getChatURL(ilChatroomObjectGUI $gui, int $scope_id=0)
$query
static string $privateRoomsAccessTable
$results
closePrivateRoom(int $id)
unbanUser($user_id)
Deletes entry from banTable matching roomId and given $user_id and returns the number of affected row...
connectUser(ilChatroomUser $user)
getUniquePrivateRoomTitle(string $title)
inviteUserToPrivateRoom(int $user_id, int $proom_id)
$comment
Definition: buildRTE.php:72
$filename
Definition: buildRTE.php:78
isAllowedToEnterPrivateRoom(int $chat_userid, int $proom_id)
Class ilChatroom.
foreach($mandatory_scripts as $file) $timestamp
Definition: buildRTE.php:70
static getInstanceByObjId(?int $obj_id, bool $stop_on_error=true)
get an instance of an Ilias object by object id
Class ilChatroomUser.
static byObjectId(int $object_id)
disconnectUser(int $user_id)
$id
plugin.php for ilComponentBuildPluginInfoObjectiveTest::testAddPlugins
Definition: plugin.php:23
$a
thx to https://mlocati.github.io/php-cs-fixer-configurator for the examples
isOwnerOfPrivateRoom(int $user_id, int $proom_id)
static string $sessionTable
$message
Definition: xapiexit.php:32
getUsername()
Returns username from Object or SESSION.
$url
static string $userTable
unsubscribeUserFromPrivateRoom(int $room_id, int $user_id)
getAccessibleRoomIdByTitleMap(int $user_id)
Fetches and returns a Array<Integer, String> of all accessible repository object chats in the main tr...
subscribeUserToPrivateRoom(int $room_id, int $user_id)
initialize(array $rowdata)
Sets $this->roomId by given array $rowdata and calls setSetting method foreach available setting in $...
userIsInPrivateRoom(int $room_id, int $user_id)
array int $roomId
saveSettings(array $settings)
static checkPermissions(int $usrId, int $refId, array $permissions)
getActivePrivateRooms(int $userid)
$i
Definition: metadata.php:41
static string $uploadTable