19 declare(strict_types=1);
37 return (
bool)
$ilSetting->get(
'session_statistics',
"1");
43 public static function createRawEntry(
string $a_session_id,
int $a_session_type,
int $a_timestamp,
int $a_user_id): void
49 if (!$a_user_id || !$a_session_id || !self::isActive()) {
57 "usr_session_stats_raw",
59 "session_id" => array(
"text", $a_session_id)
62 "type" => array(
"integer", $a_session_type),
63 "start_time" => array(
"integer", $a_timestamp),
64 "user_id" => array(
"integer", $a_user_id)
76 public static function closeRawEntry($a_session_id, ?
int $a_context = null, $a_expired_at = null): void
82 if (!self::isActive()) {
87 if (!is_array($a_session_id)) {
89 $end_time = $a_expired_at;
93 $sql =
"UPDATE usr_session_stats_raw" .
94 " SET end_time = " .
$ilDB->quote($end_time,
"integer");
96 $sql .=
",end_context = " .
$ilDB->quote($a_context,
"integer");
98 $sql .=
" WHERE session_id = " .
$ilDB->quote($a_session_id,
"text") .
99 " AND end_time IS NULL";
100 $ilDB->manipulate($sql);
103 elseif (!$a_expired_at) {
104 $sql =
"UPDATE usr_session_stats_raw" .
105 " SET end_time = " .
$ilDB->quote(time(),
"integer");
107 $sql .=
",end_context = " .
$ilDB->quote($a_context,
"integer");
109 $sql .=
" WHERE " .
$ilDB->in(
"session_id", $a_session_id,
false,
"text") .
110 " AND end_time IS NULL";
111 $ilDB->manipulate($sql);
115 foreach ($a_session_id as
$id => $ts) {
116 $sql =
"UPDATE usr_session_stats_raw" .
117 " SET end_time = " .
$ilDB->quote($ts,
"integer");
119 $sql .=
",end_context = " .
$ilDB->quote($a_context,
"integer");
121 $sql .=
" WHERE session_id = " .
$ilDB->quote(
$id,
"text") .
122 " AND end_time IS NULL";
123 $ilDB->manipulate($sql);
137 $ilDB = $DIC[
'ilDB'];
140 $sql =
"SELECT MAX(slot_end) previous_slot_end" .
141 " FROM usr_session_stats";
144 $previous_slot_end = $row[
"previous_slot_end"];
148 if (!$previous_slot_end) {
149 $slot = (
int) (floor(date(
"i") / self::SLOT_SIZE));
152 $current_slot_begin = mktime((
int) date(
"H", $a_now) - 1, 60 - self::SLOT_SIZE, 0);
156 $current_slot_begin = mktime((
int) date(
"H", $a_now), ($slot - 1) * self::SLOT_SIZE, 0);
159 $current_slot_begin = $previous_slot_end + 1;
162 $current_slot_end = $current_slot_begin + (60 * self::SLOT_SIZE) - 1;
165 if ($current_slot_end < $a_now) {
166 return array($current_slot_begin, $current_slot_end);
181 $ilDB = $DIC[
'ilDB'];
183 $sql =
"SELECT COUNT(*) counter FROM usr_session_stats_raw" .
184 " WHERE (end_time IS NULL OR end_time >= " .
$ilDB->quote($a_time,
"integer") .
")" .
185 " AND start_time <= " .
$ilDB->quote($a_time,
"integer") .
189 return (
int) $row[
"counter"];
195 protected static function getRawData(
int $a_begin,
int $a_end): array
199 $ilDB = $DIC[
'ilDB'];
201 $sql =
"SELECT start_time,end_time,end_context FROM usr_session_stats_raw" .
202 " WHERE start_time <= " .
$ilDB->quote($a_end,
"integer") .
203 " AND (end_time IS NULL OR end_time >= " .
$ilDB->quote($a_begin,
"integer") .
")" .
205 " ORDER BY start_time";
223 $ilDB = $DIC[
'ilDB'];
225 $ilAtomQuery =
$ilDB->buildAtomQuery();
226 $ilAtomQuery->addTableLock(
"usr_session_stats");
231 $slot = self::getCurrentSlot($a_now);
232 if (!is_array($slot)) {
239 "slot_begin" => array(
"integer", $slot[0]),
240 "slot_end" => array(
"integer", $slot[1]),
242 $ilDB->
insert(
"usr_session_stats", $fields);
257 if (!self::isActive()) {
261 $slot = self::createNewAggregationSlot($a_now);
262 while (is_array($slot)) {
263 self::aggregateRawHelper($slot[0], $slot[1]);
264 $slot = self::createNewAggregationSlot($a_now);
268 self::deleteAggregatedRaw($a_now);
279 $ilDB = $DIC[
'ilDB'];
291 $closed_counter = $events = array();
293 foreach (self::getRawData($a_begin, $a_end) as $item) {
302 if ($item[
"start_time"] >= $a_begin) {
304 $events[$item[
"start_time"]][] = 1;
307 if ($item[
"end_time"] && $item[
"end_time"] <= $a_end) {
308 if (in_array($item[
"end_context"], $separate_closed,
true)) {
309 if (!isset($closed_counter[$item[
"end_context"]])) {
310 $closed_counter[$item[
"end_context"]] = 0;
313 $closed_counter[$item[
"end_context"]]++;
315 $closed_counter[0] = ($closed_counter[0] ?? 0) + 1;
317 $events[$item[
"end_time"]][] = -1;
322 $active_begin = self::getNumberOfActiveRawSessions($a_begin - 1);
323 $active_end = $active_min = $active_max = $active_avg = $active_begin;
326 if (count($events)) {
327 $last_update_avg = $a_begin - 1;
328 $slot_seconds = self::SLOT_SIZE * 60;
333 foreach ($events as $ts => $actions) {
335 foreach ($actions as $action) {
347 if ($active_end > $active_max) {
348 $active_max = $active_end;
352 if ($active_end < $active_min) {
353 $active_min = $active_end;
357 $diff = $ts - $last_update_avg;
358 $active_avg += $diff / $slot_seconds * $active_end;
359 $last_update_avg = $ts;
363 if ($last_update_avg < $a_end) {
364 $diff = $a_end - $last_update_avg;
365 $active_avg += $diff / $slot_seconds * $active_end;
368 $active_avg = round($active_avg);
375 $max_sessions = self::getLimitForSlot($a_begin);
379 "active_min" => array(
"integer", $active_min),
380 "active_max" => array(
"integer", $active_max),
381 "active_avg" => array(
"integer", $active_avg),
382 "active_end" => array(
"integer", $active_end),
383 "opened" => array(
"integer", $opened_counter),
390 "closed_misc" => array(
"integer", (
int) ($closed_counter[0] ?? 0)),
391 "max_sessions" => array(
"integer", $max_sessions)
396 array(
"slot_begin" => array(
"integer", $a_begin),
397 "slot_end" => array(
"integer", $a_end))
410 $ilDB = $DIC[
'ilDB'];
413 $cut = $a_now - (60 * 60 * 24 * 7);
415 $ilDB->manipulate(
"DELETE FROM usr_session_stats_raw" .
416 " WHERE start_time <= " .
$ilDB->quote($cut,
"integer"));
426 $ilDB = $DIC[
'ilDB'];
428 $sql =
"SELECT max(slot_end) latest FROM usr_session_stats" .
429 " WHERE active_max >= max_sessions" .
430 " AND max_sessions > " .
$ilDB->quote(0,
"integer");
433 if ($row[
"latest"]) {
434 return (
int) $row[
"latest"];
448 $ilDB = $DIC[
'ilDB'];
450 $sql =
"SELECT SUM(slot_end-slot_begin) dur FROM usr_session_stats" .
451 " WHERE active_max >= max_sessions" .
452 " AND max_sessions > " .
$ilDB->quote(0,
"integer") .
453 " AND slot_end > " .
$ilDB->quote($a_from,
"integer") .
454 " AND slot_begin < " .
$ilDB->quote($a_to,
"integer");
458 return (
int) $row[
"dur"];
471 $ilDB = $DIC[
'ilDB'];
473 $sql =
"SELECT SUM(opened) opened, SUM(closed_manual) closed_manual," .
474 " SUM(closed_expire) closed_expire, SUM(closed_idle) closed_idle," .
475 " SUM(closed_idle_first) closed_idle_first, SUM(closed_limit) closed_limit," .
476 " SUM(closed_login) closed_login, SUM(closed_misc) closed_misc" .
477 " FROM usr_session_stats" .
478 " WHERE slot_end > " .
$ilDB->quote($a_from,
"integer") .
479 " AND slot_begin < " .
$ilDB->quote($a_to,
"integer");
487 public static function getActiveSessions(
int $a_from,
int $a_to): array
492 $ilDB = $DIC[
'ilDB'];
494 $sql =
"SELECT slot_begin, slot_end, active_min, active_max, active_avg," .
496 " FROM usr_session_stats" .
497 " WHERE slot_end > " .
$ilDB->quote($a_from,
"integer") .
498 " AND slot_begin < " .
$ilDB->quote($a_to,
"integer") .
499 " ORDER BY slot_begin";
504 foreach ($row as
$key => $value) {
519 $ilDB = $DIC[
'ilDB'];
521 $sql =
"SELECT max(slot_end) latest FROM usr_session_stats";
524 if ($row[
"latest"]) {
525 return (
int) $row[
"latest"];
539 $ilDB = $DIC[
'ilDB'];
543 $sql =
"SELECT maxval FROM usr_session_log" .
544 " WHERE tstamp <= " .
$ilDB->quote($a_timestamp,
"integer") .
545 " ORDER BY tstamp DESC";
548 if (isset($val[
"maxval"]) && $val[
"maxval"]) {
549 return (
int) $val[
"maxval"];
562 $ilDB = $DIC[
'ilDB'];
566 $new_value = $a_new_value;
569 if ($new_value !== $old_value) {
571 "tstamp" => array(
"timestamp", time()),
572 "maxval" => array(
"integer", $new_value),
573 "user_id" => array(
"integer",
$ilUser->getId())
575 $ilDB->insert(
"usr_session_log", $fields);
const DEFAULT_MAX_COUNT
default value for settings that have not been defined in setup or administration yet ...
static updateLimitLog(int $a_new_value)
Log max session setting.
static getNumberOfActiveRawSessions(int $a_time)
Count number of active sessions at given time.
static getLimitForSlot(int $a_timestamp)
Get max session setting for given timestamp.
static getNumberOfSessionsByType(int $a_from, int $a_to)
Get session counters by type (opened, closed)
static deleteAggregatedRaw($a_now)
Remove already aggregated raw data.
insert(string $table_name, array $values)
static createRawEntry(string $a_session_id, int $a_session_type, int $a_timestamp, int $a_user_id)
Create raw data entry.
const SESSION_CLOSE_LOGIN
static aggretateRaw(int $a_now)
Aggregate raw session data (older than given time)
const SESSION_CLOSE_EXPIRE
static aggregateRawHelper(int $a_begin, int $a_end)
Aggregate statistics data for one slot.
static getLastAggregation()
Get timestamp of last aggregation.
static getCurrentSlot(int $a_now)
Get next slot to aggregate.
static closeRawEntry($a_session_id, ?int $a_context=null, $a_expired_at=null)
Close raw data entry.
const SESSION_CLOSE_LIMIT
static getRawData(int $a_begin, int $a_end)
Read raw data for timespan.
static getMaxedOutDuration(int $a_from, int $a_to)
Get maxed out duration in given timeframe.
static isActive()
Is session statistics active at all?
static array $session_types_controlled
$id
plugin.php for ilComponentBuildPluginInfoObjectiveTest::testAddPlugins
static createNewAggregationSlot(int $a_now)
Create new slot (using table lock)
static getLastMaxedOut()
Get latest slot during which sessions were maxed out.
const SESSION_CLOSE_FIRST