19 declare(strict_types=0);
65 ?
int $parent_obj_id =
null 78 if ($parent_obj_id ==
null) {
80 'SELECT r2.obj_id par_obj_id FROM object_reference r1 ' .
81 'JOIN tree t ON t.child = r1.ref_id ' .
82 'JOIN object_reference r2 ON r2.ref_id = t.parent ' .
83 'WHERE r1.obj_id = ' .
$ilDB->quote($obj_id,
'integer')
86 while ($prec =
$ilDB->fetchAssoc($pset)) {
87 $nid =
$ilDB->nextId(
"write_event");
89 'INSERT INTO write_event ' .
90 '(write_id, obj_id, parent_obj_id, usr_id, action, ts) VALUES ' .
91 '(%s, %s, %s, %s, %s, ' .
$ilDB->now() .
')',
92 $ilDB->quote($nid,
'integer'),
93 $ilDB->quote($obj_id,
'integer'),
94 $ilDB->quote($prec[
"par_obj_id"],
'integer'),
95 $ilDB->quote($usr_id,
'integer'),
96 $ilDB->quote($action,
'text')
99 $aff =
$ilDB->manipulate($query);
102 $nid =
$ilDB->nextId(
"write_event");
104 'INSERT INTO write_event ' .
105 '(write_id, obj_id, parent_obj_id, usr_id, action, ts) ' .
106 'VALUES (%s,%s,%s,%s,%s,' .
$ilDB->now() .
')',
107 $ilDB->quote($nid,
'integer'),
108 $ilDB->quote($obj_id,
'integer'),
109 $ilDB->quote($parent_obj_id,
'integer'),
110 $ilDB->quote($usr_id,
'integer'),
111 $ilDB->quote($action,
'text')
113 $aff =
$ilDB->manipulate($query);
122 bool $isCatchupWriteEvents =
true,
128 $ilDB = $DIC[
'ilDB'];
129 $tree = $DIC[
'tree'];
134 'SELECT * FROM read_event ' .
135 'WHERE obj_id = %s ' .
137 $ilDB->quote($obj_id,
'integer'),
138 $ilDB->quote($usr_id,
'integer')
141 $row =
$ilDB->fetchObject($res);
144 if ($a_ext_rc !==
null) {
145 $read_count =
'read_count = ' .
$ilDB->quote(
149 $read_count_init = max(1, (
int) $a_ext_rc);
150 $read_count_diff = max(1, (
int) $a_ext_rc) - $row->read_count;
152 $read_count =
'read_count = read_count + 1, ';
153 $read_count_init = 1;
154 $read_count_diff = 1;
158 if ($a_ext_time !==
null) {
159 $time = (
int) $a_ext_time;
161 $time =
$ilDB->quote(
162 (time() - $row->last_access) <= $validTimeSpan
163 ? $row->spent_seconds + time() - $row->last_access
164 : $row->spent_seconds,
171 if ((time() - $row->last_access) <= $validTimeSpan) {
173 $read_count_init = 1;
174 $read_count_diff = 0;
177 $time_diff = $time - (
int) ($row->spent_seconds ?? 0);
181 'UPDATE read_event SET ' .
183 'spent_seconds = %s, ' .
184 'last_access = %s ' .
185 'WHERE obj_id = %s ' .
188 $ilDB->quote(time(),
'integer'),
189 $ilDB->quote($obj_id,
'integer'),
190 $ilDB->quote($usr_id,
'integer')
192 $aff =
$ilDB->manipulate($query);
194 self::_recordObjStats($obj_id, $time_diff, $read_count_diff);
196 if ($a_ext_time !==
false) {
197 $time = (
int) $a_ext_time;
202 $time_diff = $time - (
int) ($row->spent_seconds ?? 0);
208 'obj_id' => array(
'integer', $obj_id),
209 'usr_id' => array(
'integer', $usr_id)
212 'read_count' => array(
'integer', $read_count_init),
213 'spent_seconds' => array(
'integer', $time),
214 'first_access' => array(
'timestamp', date(
"Y-m-d H:i:s")),
216 'last_access' => array(
'integer', time())
220 self::$has_accessed[$obj_id][$usr_id] =
true;
222 self::_recordObjStats($obj_id, $time_diff, $read_count_diff);
225 if ($isCatchupWriteEvents) {
230 if (!in_array($a_type, array(
"cat",
"root",
"crs"))) {
231 if ($tree->isInTree($a_ref_id)) {
232 $path = $tree->getPathId($a_ref_id);
234 foreach (
$path as $p) {
238 if (($p != $a_ref_id) && (in_array(
247 'SELECT * FROM read_event ' .
248 'WHERE obj_id = %s ' .
250 $ilDB->quote($obj2_id,
'integer'),
251 $ilDB->quote($usr_id,
'integer')
253 $res2 =
$ilDB->query($query);
254 if ($row2 =
$ilDB->fetchAssoc($res2)) {
258 'UPDATE read_event SET ' .
259 'childs_read_count = childs_read_count + %s ,' .
260 'childs_spent_seconds = childs_spent_seconds + %s ' .
261 'WHERE obj_id = %s ' .
263 $ilDB->quote((
int) $read_count_diff,
'integer'),
264 $ilDB->quote((
int) $time_diff,
'integer'),
265 $ilDB->quote($obj2_id,
'integer'),
266 $ilDB->quote($usr_id,
'integer')
268 $aff =
$ilDB->manipulate($query);
270 self::_recordObjStats(
275 (
int) $read_count_diff
282 'obj_id' => array(
'integer', $obj2_id),
283 'usr_id' => array(
'integer', $usr_id)
286 'read_count' => array(
'integer', 1),
287 'spent_seconds' => array(
'integer', $time),
288 'first_access' => array(
'timestamp',
291 'last_access' => array(
'integer', time()),
292 'childs_read_count' => array(
'integer',
293 (
int) $read_count_diff
295 'childs_spent_seconds' => array(
'integer',
301 self::$has_accessed[$obj2_id][$usr_id] =
true;
303 self::_recordObjStats(
308 (
int) $read_count_diff
324 ?
int $a_spent_seconds,
326 ?
int $a_childs_spent_seconds =
null,
327 ?
int $a_child_read_count =
null 331 $ilDB = $DIC[
'ilDB'];
341 $fields[
'log_id'] = array(
"integer",
$ilDB->nextId(
'obj_stat_log'));
342 $fields[
"obj_id"] = array(
"integer", $a_obj_id);
344 $fields[
"tstamp"] = array(
"timestamp", $now);
345 $fields[
"yyyy"] = array(
"integer", date(
"Y"));
346 $fields[
"mm"] = array(
"integer", date(
"m"));
347 $fields[
"dd"] = array(
"integer", date(
"d"));
348 $fields[
"hh"] = array(
"integer", date(
"H"));
349 if ($a_spent_seconds > 0) {
350 $fields[
"spent_seconds"] = array(
"integer", $a_spent_seconds);
352 if ($a_read_count > 0) {
353 $fields[
"read_count"] = array(
"integer", $a_read_count);
355 if ($a_childs_spent_seconds > 0) {
356 $fields[
"childs_spent_seconds"] = array(
"integer",
357 $a_childs_spent_seconds
360 if ($a_child_read_count > 0) {
361 $fields[
"childs_read_count"] = array(
"integer",
365 $ilDB->insert(
"obj_stat_log", $fields);
368 if (mt_rand(1, 100) == 1) {
369 self::_syncObjectStats($now);
375 int $a_minimum = 20000
379 $ilDB = $DIC[
'ilDB'];
388 $set =
$ilDB->query(
"SELECT COUNT(*) AS counter FROM obj_stat_log");
389 $row =
$ilDB->fetchAssoc($set);
390 if ($row[
"counter"] >= $a_minimum) {
391 $ilAtomQuery =
$ilDB->buildAtomQuery();
392 $ilAtomQuery->addTableLock(
'obj_stat_log');
393 $ilAtomQuery->addTableLock(
'obj_stat_tmp');
395 $ilAtomQuery->addQueryCallable(
400 "SELECT COUNT(*) AS counter FROM obj_stat_log" 403 if ($row[
"counter"] >= $a_minimum) {
406 "INSERT INTO obj_stat_tmp" .
407 " SELECT * FROM obj_stat_log" .
408 " WHERE tstamp < " . $ilDB->
quote(
416 "DELETE FROM obj_stat_log" .
417 " WHERE tstamp < " . $ilDB->
quote(
435 $ilAtomQuery->addTableLock(
'obj_stat_tmp');
436 $ilAtomQuery->addTableLock(
'obj_stat');
438 $ilAtomQuery->addQueryCallable(
441 $sql =
"SELECT obj_id, obj_type, yyyy, mm, dd, hh, SUM(read_count) AS read_count," .
442 " SUM(childs_read_count) AS childs_read_count, SUM(spent_seconds) AS spent_seconds," .
443 " SUM(childs_spent_seconds) AS childs_spent_seconds" .
444 " FROM obj_stat_tmp" .
445 " GROUP BY obj_id, obj_type, yyyy, mm, dd, hh";
446 $set = $ilDB->
query($sql);
449 $where = array(
"obj_id" => array(
"integer",
452 "obj_type" => array(
"text",
455 "yyyy" => array(
"integer",
458 "mm" => array(
"integer", $row[
"mm"]),
459 "dd" => array(
"integer", $row[
"dd"]),
460 "hh" => array(
"integer", $row[
"hh"])
463 $where_sql = array();
464 foreach ($where as $field => $def) {
465 $where_sql[] = $field .
" = " . $ilDB->
quote(
470 $where_sql = implode(
" AND ", $where_sql);
474 "SELECT read_count, childs_read_count, spent_seconds," .
475 "childs_spent_seconds" .
477 " WHERE " . $where_sql
483 $fields = array(
"read_count" => array(
"integer",
484 $old[
"read_count"] + $row[
"read_count"]
486 "childs_read_count" => array(
"integer",
487 $old[
"childs_read_count"] + $row[
"childs_read_count"]
489 "spent_seconds" => array(
"integer",
490 $old[
"spent_seconds"] + $row[
"spent_seconds"]
492 "childs_spent_seconds" => array(
"integer",
493 $old[
"childs_spent_seconds"] + $row[
"childs_spent_seconds"]
497 $ilDB->
update(
"obj_stat", $fields, $where);
501 $fields[
"read_count"] = array(
"integer",
504 $fields[
"childs_read_count"] = array(
"integer",
505 $row[
"childs_read_count"]
507 $fields[
"spent_seconds"] = array(
"integer",
508 $row[
"spent_seconds"]
510 $fields[
"childs_spent_seconds"] = array(
"integer",
511 $row[
"childs_spent_seconds"]
514 $ilDB->
insert(
"obj_stat", $fields);
519 $ilDB->
query(
"DELETE FROM obj_stat_tmp");
543 $ilDB = $DIC[
'ilDB'];
545 $query =
"SELECT obj_id FROM catch_write_events " .
546 "WHERE obj_id = " .
$ilDB->quote($obj_id,
'integer') .
" " .
547 "AND usr_id = " .
$ilDB->quote($usr_id,
'integer');
549 if (
$res->numRows()) {
559 "catch_write_events",
561 "obj_id" => array(
"integer", $obj_id),
562 "usr_id" => array(
"integer", $usr_id)
565 "ts" => array(
"timestamp", $ts)
583 $ilDB = $DIC[
'ilDB'];
585 "FROM catch_write_events " .
586 "WHERE obj_id=" .
$ilDB->quote($obj_id,
'integer') .
" " .
587 "AND usr_id=" .
$ilDB->quote($usr_id,
'integer');
591 $catchup = $row[
'ts'];
594 if ($catchup ==
null) {
596 'SELECT * FROM write_event ' .
597 'WHERE obj_id = %s ' .
598 'AND usr_id <> %s ' .
600 $ilDB->quote($obj_id,
'integer'),
601 $ilDB->quote($usr_id,
'integer')
606 'SELECT * FROM write_event ' .
607 'WHERE obj_id = %s ' .
608 'AND usr_id <> %s ' .
611 $ilDB->quote($obj_id,
'integer'),
612 $ilDB->quote($usr_id,
'integer'),
613 $ilDB->quote($catchup,
'timestamp')
637 $ilDB = $DIC[
'ilDB'];
640 "FROM catch_write_events " .
641 "WHERE obj_id=" .
$ilDB->quote($obj_id,
'integer') .
" " .
642 "AND usr_id=" .
$ilDB->quote($usr_id,
'integer');
646 $catchup = $row[
'ts'];
649 if ($catchup ==
null) {
652 'SELECT * FROM write_event ' .
653 'WHERE obj_id = %s ' .
655 $ilDB->quote($obj_id,
'integer'),
656 $ilDB->quote($usr_id,
'integer')
662 'SELECT * FROM write_event ' .
663 'WHERE obj_id = %s ' .
664 'AND usr_id <> %s ' .
666 $ilDB->quote($obj_id,
'integer'),
667 $ilDB->quote($usr_id,
'integer'),
668 $ilDB->quote($catchup,
'timestamp')
673 $numRows =
$res->numRows();
678 return ($catchup ==
null) ? 1 : 2;
693 $ilDB = $DIC[
'ilDB'];
695 if ($usr_id ==
null) {
697 'SELECT * FROM read_event ' .
698 'WHERE obj_id = %s ' .
699 'ORDER BY last_access DESC',
700 $ilDB->quote($obj_id,
'integer')
705 'SELECT * FROM read_event ' .
706 'WHERE obj_id = %s ' .
708 'ORDER BY last_access DESC',
709 $ilDB->quote($obj_id,
'integer'),
710 $ilDB->quote($usr_id,
'integer')
718 $events[$counter][
'obj_id'] = $row[
'obj_id'];
719 $events[$counter][
'usr_id'] = $row[
'usr_id'];
720 $events[$counter][
'last_access'] = $row[
'last_access'];
721 $events[$counter][
'read_count'] = $row[
'read_count'];
722 $events[$counter][
'spent_seconds'] = $row[
'spent_seconds'];
723 $events[$counter][
'first_access'] = $row[
'first_access'];
734 $ilDB = $DIC[
'ilDB'];
737 'SELECT DISTINCT(usr_id) usr FROM read_event ' .
738 'WHERE obj_id = %s ',
739 $ilDB->quote($a_obj_id,
'integer')
743 while ($row =
$ilDB->fetchObject($res)) {
744 $users[] = (
int) $row->usr;
752 public static function hasAccessed(
int $a_obj_id,
int $a_usr_id): bool
756 $ilDB = $DIC[
'ilDB'];
758 if (isset(self::$has_accessed[$a_obj_id][$a_usr_id])) {
759 return self::$has_accessed[$a_obj_id][$a_usr_id];
763 "SELECT usr_id FROM read_event WHERE " .
764 "obj_id = " .
$ilDB->quote($a_obj_id,
"integer") .
" AND " .
765 "usr_id = " .
$ilDB->quote($a_usr_id,
"integer")
767 if ($rec =
$ilDB->fetchAssoc($set)) {
768 return self::$has_accessed[$a_obj_id][$a_usr_id] =
true;
770 return self::$has_accessed[$a_obj_id][$a_usr_id] =
false;
783 $ilDB = $DIC[
'ilDB'];
793 'SELECT r1.obj_id,r2.obj_id p,d.owner,%s,d.create_date ' .
794 'FROM object_data d ' .
795 'LEFT JOIN write_event w ON d.obj_id = w.obj_id ' .
796 'JOIN object_reference r1 ON d.obj_id=r1.obj_id ' .
797 'JOIN tree t ON t.child=r1.ref_id ' .
798 'JOIN object_reference r2 on r2.ref_id=t.parent ' .
799 'WHERE w.obj_id IS NULL',
800 $ilDB->quote(
'create',
'text')
804 while ($rec =
$ilDB->fetchAssoc($set)) {
805 $nid =
$ilDB->nextId(
"write_event");
806 $query =
'INSERT INTO write_event ' .
807 '(write_id, obj_id,parent_obj_id,usr_id,action,ts) VALUES (' .
808 $ilDB->quote($nid,
"integer") .
"," .
809 $ilDB->quote($rec[
"obj_id"],
"integer") .
"," .
810 $ilDB->quote($rec[
"p"],
"integer") .
"," .
811 $ilDB->quote($rec[
"owner"],
"integer") .
"," .
812 $ilDB->quote(
"create",
"text") .
"," .
813 $ilDB->quote($rec[
"create_date"],
"timestamp") .
821 $ilSetting->set(
'enable_change_event_tracking',
'1');
835 $ilSetting->set(
'enable_change_event_tracking',
'0');
847 return $ilSetting->get(
'enable_change_event_tracking',
'0') ==
'1';
853 public static function _delete(
int $a_obj_id): bool
857 $ilDB = $DIC[
'ilDB'];
859 'DELETE FROM write_event WHERE obj_id = %s ',
860 $ilDB->quote($a_obj_id,
'integer')
862 $aff =
$ilDB->manipulate($query);
865 'DELETE FROM read_event WHERE obj_id = %s ',
866 $ilDB->quote($a_obj_id,
'integer')
868 $aff =
$ilDB->manipulate($query);
876 $ilDB = $DIC[
'ilDB'];
879 "DELETE FROM read_event" .
880 " WHERE obj_id = " .
$ilDB->quote($a_obj_id,
"integer")
890 $ilDB = $DIC[
'ilDB'];
893 "DELETE FROM read_event" .
894 " WHERE obj_id = " .
$ilDB->quote($a_obj_id,
"integer") .
895 " AND " .
$ilDB->in(
"usr_id", $a_user_ids,
"",
"integer")
903 $ilDB = $DIC[
'ilDB'];
906 "SELECT usr_id FROM read_event" .
907 " WHERE obj_id = " .
$ilDB->quote($a_obj_id,
"integer")
909 while ($row =
$ilDB->fetchAssoc($set)) {
924 string $t_first_access
928 $ilDB = $DIC->database();
930 'UPDATE read_event SET first_access=%s, last_access = %s WHERE obj_id=%s AND usr_id=%s',
931 array(
'timestamp',
'integer',
'integer',
'integer'),
932 array($t_first_access, $i_last_access, $obj_id, $usr_id)
static _delete(int $a_obj_id)
Delete object entries.
static _activate()
Activates change event tracking.
numRows(ilDBStatement $statement)
insert(string $table_name, array $values)
static _enabledObjectStatistics()
static _lookupUncaughtWriteEvents(int $obj_id, int $usr_id)
Reads all write events which occured on the object which happened after the last time the user caught...
fetchAssoc(ilDBStatement $statement)
update(string $table_name, array $values, array $where)
$where MUST contain existing columns only.
static hasAccessed(int $a_obj_id, int $a_usr_id)
Has accessed.
quote($value, string $type)
static _updateAccessForScormOfflinePlayer(int $obj_id, int $usr_id, int $i_last_access, string $t_first_access)
_updateAccessForScormOfflinePlayer needed to synchronize last_access and first_access when learning m...
static now()
Return current timestamp in Y-m-d H:i:s format.
static _syncObjectStats(?int $a_now=null, int $a_minimum=20000)
static _lookupObjId(int $ref_id)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
static lookupUsersInProgress(int $a_obj_id)
static _recordReadEvent(string $a_type, int $a_ref_id, int $obj_id, int $usr_id, bool $isCatchupWriteEvents=true, $a_ext_rc=null, $a_ext_time=null)
static _lookupChangeState(int $obj_id, int $usr_id)
Returns the change state of the object for the specified user.
static _recordWriteEvent(int $obj_id, int $usr_id, string $action, ?int $parent_obj_id=null)
Records a write event.
query(string $query)
Run a (read-only) Query on the database.
static _deleteReadEvents(int $a_obj_id)
static _lookupReadEvents($obj_id, $usr_id=null)
Reads all read events which occured on the object.
foreach($mandatory_scripts as $file) $timestamp
Class ilChangeEvent tracks change events on repository objects.
static _recordObjStats(int $a_obj_id, ?int $a_spent_seconds, ?int $a_read_count, ?int $a_childs_spent_seconds=null, ?int $a_child_read_count=null)
static _isActive()
Returns true, if change event tracking is active.
static _deactivate()
Deactivates change event tracking.
static _deleteReadEventsForUsers(int $a_obj_id, array $a_user_ids)
static array $has_accessed
static _lookupType(int $id, bool $reference=false)
static _catchupWriteEvents(int $obj_id, int $usr_id, ?string $timestamp=null)
Catches up with all write events which occured before the specified timestamp.
static _getAllUserIds(int $a_obj_id)
static _getValidTimeSpan()