ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilUserQuery.php
Go to the documentation of this file.
1 <?php
2 
25 {
26  public const DEFAULT_ORDER_FIELD = 'login';
27 
28  private string $order_field = self::DEFAULT_ORDER_FIELD;
29  private string $order_dir = 'asc';
30  private int $offset = 0;
31  private int $limit = 50;
32  private string $text_filter = '';
33  private string $activation = '';
35  private bool $limited_access = false;
36  private bool $no_courses = false;
37  private bool $no_groups = false;
38  private int $crs_grp = 0;
39  private int $role = 0;
40  private ?array $user_folder = null; // Missing array type.
41  private array $additional_fields = []; // Missing array type.
42  private array $users = []; // Missing array type.
43  private string $first_letter = '';
44  private bool $has_access = false;
45  private string $authentication_method = '';
46  protected array $udf_filter = []; // Missing array type.
48  private array $default_fields = [
49  "usr_id",
50  "login",
51  "firstname",
52  "lastname",
53  "email",
54  "second_email",
55  "time_limit_until",
56  "time_limit_unlimited",
57  "time_limit_owner",
58  "last_login",
59  "active"
60  ];
61 
62  public function __construct()
63  {
64  }
65 
70  public function setUdfFilter(array $a_val): void // Missing array type.
71  {
72  $valid_udfs = [];
73 
74  $definitions = \ilUserDefinedFields::_getInstance()->getDefinitions();
75  foreach ($a_val as $udf_name => $udf_value) {
76  [$udf_string, $udf_id] = explode('_', $udf_name);
77  if (array_key_exists((int) $udf_id, $definitions)) {
78  $valid_udfs[$udf_name] = $udf_value;
79  }
80  }
81  $this->udf_filter = $valid_udfs;
82  }
83 
88  public function getUdfFilter(): array // Missing array type.
89  {
90  return $this->udf_filter;
91  }
92 
97  public function setOrderField(string $a_order): void
98  {
99  $this->order_field = $a_order;
100  }
101 
107  public function setOrderDirection(string $a_dir): void
108  {
109  $this->order_dir = $a_dir;
110  }
111 
112  public function setOffset(int $a_offset): void
113  {
114  $this->offset = $a_offset;
115  }
116 
117  public function setLimit(int $a_limit): void
118  {
119  $this->limit = $a_limit;
120  }
121 
125  public function setTextFilter(string $a_filter): void
126  {
127  $this->text_filter = $a_filter;
128  }
129 
134  public function setActionFilter(string $a_activation): void
135  {
136  $this->activation = $a_activation;
137  }
138 
142  public function setLastLogin(?ilDateTime $dt = null): void
143  {
144  $this->last_login = $dt;
145  }
146 
150  public function setLimitedAccessFilter(bool $a_status): void
151  {
152  $this->limited_access = $a_status;
153  }
154 
155  public function setNoCourseFilter(bool $a_no_course): void
156  {
157  $this->no_courses = $a_no_course;
158  }
159 
160  public function setNoGroupFilter(bool $a_no_group): void
161  {
162  $this->no_groups = $a_no_group;
163  }
164 
169  public function setCourseGroupFilter(int $a_cg_id): void
170  {
171  $this->crs_grp = $a_cg_id;
172  }
173 
178  public function setRoleFilter(int $a_role_id): void
179  {
180  $this->role = $a_role_id;
181  }
182 
187  public function setUserFolder(?array $a_fold_id): void // Missing array type.
188  {
189  $this->user_folder = $a_fold_id;
190  }
191 
195  public function setAdditionalFields(array $a_add): void // Missing array type.
196  {
197  $this->additional_fields = $a_add;
198  }
199 
203  public function setUserFilter(array $a_filter): void // Missing array type.
204  {
205  $this->users = $a_filter;
206  }
207 
211  public function setFirstLetterLastname(string $a_fll): void
212  {
213  $this->first_letter = $a_fll;
214  }
215 
219  public function setAccessFilter(bool $a_access): void
220  {
221  $this->has_access = $a_access;
222  }
223 
228  public function setAuthenticationFilter(string $a_authentication): void
229  {
230  $this->authentication_method = $a_authentication;
231  }
232 
237  public function query(): array
238  {
239  global $DIC;
240 
241  $ilDB = $DIC['ilDB'];
242 
243 
244  $udf_fields = [];
245  $usr_ids = [];
246 
247  $join = "";
248 
249  if (is_array($this->additional_fields)) {
250  foreach ($this->additional_fields as $f) {
251  if (!in_array($f, $this->default_fields)) {
252  if ($f === "online_time") {
253  $this->default_fields[] = "ut_online.online_time";
254  $join = " LEFT JOIN ut_online ON (usr_data.usr_id = ut_online.usr_id) ";
255  } elseif ($f === 'dpro_agreed_on') {
256  $this->default_fields[] = 'dpro.dpro_agreed_on';
257  $join = ' LEFT JOIN (SELECT value AS dpro_agreed_on, usr_id FROM usr_pref WHERE keyword = "dpro_agree_date") AS dpro' .
258  ' ON (usr_data.usr_id = dpro.usr_id)';
259  } elseif (substr($f, 0, 4) === "udf_") {
260  $udf_fields[] = (int) substr($f, 4);
261  } else {
262  $this->default_fields[] = $f;
263  }
264  }
265  }
266  }
267 
268  // if udf fields are involved we need the definitions
269  $udf_def = [];
270  if (count($udf_fields) > 0) {
271  $udf_def = ilUserDefinedFields::_getInstance()->getDefinitions();
272  }
273 
274  // join udf table
275  foreach ($udf_fields as $id) {
276  $udf_table = ($udf_def[$id]["field_type"] != UDF_TYPE_WYSIWYG)
277  ? "udf_text"
278  : "udf_clob";
279  $join .= " LEFT JOIN " . $udf_table . " ud_" . $id . " ON (ud_" . $id . ".field_id=" . $ilDB->quote($id) . " AND ud_" . $id . ".usr_id = usr_data.usr_id) ";
280  }
281 
282  // count query
283  $count_query = "SELECT count(usr_data.usr_id) cnt" .
284  " FROM usr_data";
285 
286  $all_multi_fields = ["interests_general", "interests_help_offered", "interests_help_looking"];
287  $multi_fields = [];
288 
289  $sql_fields = [];
290  foreach ($this->default_fields as $idx => $field) {
291  if (!$field) {
292  continue;
293  }
294 
295  if (in_array($field, $all_multi_fields)) {
296  $multi_fields[] = $field;
297  } elseif (strpos($field, ".") === false) {
298  $sql_fields[] = "usr_data." . $field;
299  } else {
300  $sql_fields[] = $field;
301  }
302  }
303 
304  // udf fields
305  foreach ($udf_fields as $id) {
306  $sql_fields[] = "ud_" . $id . ".value udf_" . $id;
307  }
308 
309  // basic query
310  $query = "SELECT " . implode(",", $sql_fields) .
311  " FROM usr_data" .
312  $join;
313 
314  $count_query .= " " . $join;
315 
316  // filter
317  $query .= " WHERE usr_data.usr_id <> " . $ilDB->quote(ANONYMOUS_USER_ID, "integer");
318 
319  // User filter
320  $count_query .= " WHERE 1 = 1 ";
321  $count_user_filter = "usr_data.usr_id != " . $ilDB->quote(ANONYMOUS_USER_ID, "integer");
322  if ($this->users && is_array(($this->users))) {
323  $query .= ' AND ' . $ilDB->in('usr_data.usr_id', $this->users, false, 'integer');
324  $count_user_filter = $ilDB->in('usr_data.usr_id', $this->users, false, 'integer');
325  }
326 
327  $count_query .= " AND " . $count_user_filter . " ";
328  $where = " AND";
329 
330  if ($this->first_letter != "") {
331  $add = $where . " (" . $ilDB->upper($ilDB->substr("usr_data.lastname", 1, 1)) . " = " . $ilDB->upper($ilDB->quote($this->first_letter, "text")) . ") ";
332  $query .= $add;
333  $count_query .= $add;
334  $where = " AND";
335  }
336 
337  if ($this->text_filter != "") { // email, name, login
338  $add = $where . " (" . $ilDB->like("usr_data.login", "text", "%" . $this->text_filter . "%") . " " .
339  "OR " . $ilDB->like("usr_data.firstname", "text", "%" . $this->text_filter . "%") . " " .
340  "OR " . $ilDB->like("usr_data.lastname", "text", "%" . $this->text_filter . "%") . " " .
341  "OR " . $ilDB->like("usr_data.second_email", "text", "%" . $this->text_filter . "%") . " " .
342  "OR " . $ilDB->like("usr_data.email", "text", "%" . $this->text_filter . "%") . ") ";
343  $query .= $add;
344  $count_query .= $add;
345  $where = " AND";
346  }
347 
348  if ($this->activation != "") { // activation
349  if ($this->activation === "inactive") {
350  $add = $where . " usr_data.active = " . $ilDB->quote(0, "integer") . " ";
351  } else {
352  $add = $where . " usr_data.active = " . $ilDB->quote(1, "integer") . " ";
353  }
354  $query .= $add;
355  $count_query .= $add;
356  $where = " AND";
357  }
358 
359  if ($this->last_login instanceof ilDateTime) { // last login
360  if (ilDateTime::_before($this->last_login, new ilDateTime(time() + (60 * 60 * 24), IL_CAL_UNIX), IL_CAL_DAY)) {
361  $add = $where . " usr_data.last_login < " .
362  $ilDB->quote($this->last_login->get(IL_CAL_DATETIME), "timestamp");
363  $query .= $add;
364  $count_query .= $add;
365  $where = " AND";
366  }
367  }
368  if ($this->limited_access) { // limited access
369  $add = $where . " usr_data.time_limit_unlimited= " . $ilDB->quote(0, "integer");
370  $query .= $add;
371  $count_query .= $add;
372  $where = " AND";
373  }
374 
375  // udf filter
376  foreach ($this->getUdfFilter() as $k => $f) {
377  if ($f != "") {
378  $udf_id = explode("_", $k)[1];
379  if ($udf_def[$udf_id]["field_type"] == UDF_TYPE_TEXT) {
380  $add = $where . " " . $ilDB->like("ud_" . $udf_id . ".value", "text", "%" . $f . "%");
381  } else {
382  $add = $where . " ud_" . $udf_id . ".value = " . $ilDB->quote($f, "text");
383  }
384  $query .= $add;
385  $count_query .= $add;
386  $where = " AND";
387  }
388  }
389 
390  if ($this->has_access) { //user is limited but has access
391  $unlimited = "time_limit_unlimited = " . $ilDB->quote(1, 'integer');
392  $from = "time_limit_from < " . $ilDB->quote(time(), 'integer');
393  $until = "time_limit_until > " . $ilDB->quote(time(), 'integer');
394 
395  $add = $where . ' (' . $unlimited . ' OR (' . $from . ' AND ' . $until . '))';
396  $query .= $add;
397  $count_query .= $add;
398  $where = " AND";
399  }
400  if ($this->no_courses) { // no courses assigned
401  $add = $where . " usr_data.usr_id NOT IN (" .
402  "SELECT DISTINCT ud.usr_id " .
403  "FROM usr_data ud join rbac_ua ON (ud.usr_id = rbac_ua.usr_id) " .
404  "JOIN object_data od ON (rbac_ua.rol_id = od.obj_id) " .
405  "JOIN rbac_fa ON (rbac_ua.rol_id = rbac_fa.rol_id) " .
406  "JOIN tree ON (rbac_fa.parent = tree.child) " .
407  "WHERE od.title LIKE 'il_crs_%' " .
408  "AND rbac_fa.assign = 'y' " .
409  "AND tree.tree > 0)";
410  $query .= $add;
411  $count_query .= $add;
412  $where = " AND";
413  }
414  if ($this->no_groups) { // no groups assigned
415  $add = $where . " usr_data.usr_id NOT IN (" .
416  "SELECT DISTINCT ud.usr_id " .
417  "FROM usr_data ud join rbac_ua ON (ud.usr_id = rbac_ua.usr_id) " .
418  "JOIN object_data od ON (rbac_ua.rol_id = od.obj_id) " .
419  "JOIN rbac_fa ON (rbac_ua.rol_id = rbac_fa.rol_id) " .
420  "JOIN tree ON (rbac_fa.parent = tree.child) " .
421  "WHERE od.title LIKE 'il_grp_%' " .
422  "AND rbac_fa.assign = 'y' " .
423  "AND tree.tree > 0)";
424  $query .= $add;
425  $count_query .= $add;
426  $where = " AND";
427  }
428  if ($this->crs_grp > 0) { // members of course/group
429  $cgtype = ilObject::_lookupType($this->crs_grp, true);
430  $add = $where . " usr_data.usr_id IN (" .
431  "SELECT DISTINCT ud.usr_id " .
432  "FROM usr_data ud join rbac_ua ON (ud.usr_id = rbac_ua.usr_id) " .
433  "JOIN object_data od ON (rbac_ua.rol_id = od.obj_id) " .
434  "WHERE od.title = " . $ilDB->quote("il_" . $cgtype . "_member_" . $this->crs_grp, "text") . ")";
435  $query .= $add;
436  $count_query .= $add;
437  $where = " AND";
438  }
439  if ($this->role > 0) { // global role
440  $add = $where . " usr_data.usr_id IN (" .
441  "SELECT DISTINCT ud.usr_id " .
442  "FROM usr_data ud join rbac_ua ON (ud.usr_id = rbac_ua.usr_id) " .
443  "WHERE rbac_ua.rol_id = " . $ilDB->quote($this->role, "integer") . ")";
444  $query .= $add;
445  $count_query .= $add;
446  $where = " AND";
447  }
448 
449  if ($this->user_folder) {
450  $add = $where . " " . $ilDB->in('usr_data.time_limit_owner', $this->user_folder, false, 'integer');
451  $query .= $add;
452  $count_query .= $add;
453  $where = " AND";
454  }
455 
456  if ($this->authentication_method != "") { // authentication
457  $add = $where . " usr_data.auth_mode = " . $ilDB->quote($this->authentication_method, "text") . " ";
458  $query .= $add;
459  $count_query .= $add;
460  $where = " AND";
461  }
462 
463  // order by
464  switch ($this->order_field) {
465  case "access_until":
466  if ($this->order_dir === "desc") {
467  $query .= " ORDER BY usr_data.active DESC, usr_data.time_limit_unlimited DESC, usr_data.time_limit_until DESC";
468  } else {
469  $query .= " ORDER BY usr_data.active ASC, usr_data.time_limit_unlimited ASC, usr_data.time_limit_until ASC";
470  }
471  break;
472 
473  case "online_time":
474  if ($this->order_dir === "desc") {
475  $query .= " ORDER BY ut_online.online_time DESC";
476  } else {
477  $query .= " ORDER BY ut_online.online_time ASC";
478  }
479  break;
480 
481  default:
482  if ($this->order_dir !== "asc" && $this->order_dir !== "desc") {
483  $this->order_dir = "asc";
484  }
485  if (substr($this->order_field, 0, 4) === "udf_") {
486  // #25311 check if order field is in field list
487  if (is_array($this->getUdfFilter()) && array_key_exists($this->order_field, $this->getUdfFilter())) {
488  $query .= " ORDER BY ud_" . ((int) substr($this->order_field, 4)) . ".value " . strtoupper($this->order_dir);
489  } else {
490  $query .= ' ORDER BY ' . self::DEFAULT_ORDER_FIELD . ' ' . strtoupper($this->order_dir);
491  }
492  } else {
493  if (!in_array($this->order_field, $this->default_fields)) {
494  $this->order_field = "login";
495  }
496  $query .= " ORDER BY usr_data." . $this->order_field . " " . strtoupper($this->order_dir);
497  }
498  break;
499  }
500 
501  // count query
502  $set = $ilDB->query($count_query);
503  $cnt = 0;
504  if ($rec = $ilDB->fetchAssoc($set)) {
505  $cnt = $rec["cnt"];
506  }
507 
508  $offset = $this->offset;
509  $limit = $this->limit;
510 
511  // #9866: validate offset against rowcount
512  if ($offset >= $cnt) {
513  $offset = 0;
514  }
515 
516  $ilDB->setLimit($limit, $offset);
517 
518  if (count($multi_fields)) {
519  $usr_ids = [];
520  }
521 
522  // set query
523  $set = $ilDB->query($query);
524  $result = [];
525 
526  while ($rec = $ilDB->fetchAssoc($set)) {
527  $result[] = $rec;
528  if (count($multi_fields)) {
529  $usr_ids[] = (int) $rec["usr_id"];
530  }
531  }
532 
533  // add multi-field-values to user-data
534  if (count($multi_fields) && count($usr_ids)) {
535  $usr_multi = [];
536  $set = $ilDB->query("SELECT * FROM usr_data_multi" .
537  " WHERE " . $ilDB->in("usr_id", $usr_ids, "", "integer"));
538  while ($row = $ilDB->fetchAssoc($set)) {
539  $usr_multi[(int) $row["usr_id"]][$row["field_id"]][] = $row["value"];
540  }
541  foreach ($result as $idx => $item) {
542  if (isset($usr_multi[$item["usr_id"]])) {
543  $result[$idx] = array_merge($item, $usr_multi[(int) $item["usr_id"]]);
544  }
545  }
546  }
547  return ["cnt" => $cnt, "set" => $result];
548  }
549 
550 
555  public static function getUserListData(
556  string $a_order_field,
557  string $a_order_dir,
558  int $a_offset,
559  int $a_limit,
560  string $a_string_filter = "",
561  string $a_activation_filter = "",
562  ?ilDateTime $a_last_login_filter = null,
563  bool $a_limited_access_filter = false,
564  bool $a_no_courses_filter = false,
565  int $a_course_group_filter = 0,
566  int $a_role_filter = 0,
567  ?array $a_user_folder_filter = null,
568  ?array $a_additional_fields = null,
569  ?array $a_user_filter = null,
570  string $a_first_letter = "",
571  string $a_authentication_filter = ""
572  ): array {
573  $query = new ilUserQuery();
574  $query->setOrderField($a_order_field);
575  $query->setOrderDirection($a_order_dir);
576  $query->setOffset($a_offset);
577  $query->setLimit($a_limit);
578  $query->setTextFilter($a_string_filter);
579  $query->setActionFilter($a_activation_filter);
580  $query->setLastLogin($a_last_login_filter);
581  $query->setLimitedAccessFilter($a_limited_access_filter);
582  $query->setNoCourseFilter($a_no_courses_filter);
583  $query->setCourseGroupFilter($a_course_group_filter);
584  $query->setRoleFilter($a_role_filter);
585  $query->setUserFolder($a_user_folder_filter);
586  $query->setAdditionalFields($a_additional_fields ?? []);
587  $query->setUserFilter($a_user_filter ?? []);
588  $query->setFirstLetterLastname($a_first_letter);
589  $query->setAuthenticationFilter($a_authentication_filter);
590  return $query->query();
591  }
592 }
setOrderField(string $a_order)
Set order field (column in usr_data) Default order is &#39;login&#39;.
setNoCourseFilter(bool $a_no_course)
setRoleFilter(int $a_role_id)
Set role filter obj_id of role.
setLimit(int $a_limit)
getUdfFilter()
Get udf filter.
setLimitedAccessFilter(bool $a_status)
Enable limited access filter.
string $authentication_method
const IL_CAL_DATETIME
const ANONYMOUS_USER_ID
Definition: constants.php:27
setAdditionalFields(array $a_add)
Set additional fields (columns in usr_data or &#39;online_time&#39;)
setOffset(int $a_offset)
static _before(ilDateTime $start, ilDateTime $end, string $a_compare_field='', string $a_tz='')
compare two dates and check start is before end This method does not consider tz offsets.
setFirstLetterLastname(string $a_fll)
set first letter lastname filter
setTextFilter(string $a_filter)
Text (like) filter in login, firstname, lastname or email.
ilDateTime $last_login
const IL_CAL_UNIX
setLastLogin(?ilDateTime $dt=null)
Set last login filter.
query()
Query usr_data.
static getUserListData(string $a_order_field, string $a_order_dir, int $a_offset, int $a_limit, string $a_string_filter="", string $a_activation_filter="", ?ilDateTime $a_last_login_filter=null, bool $a_limited_access_filter=false, bool $a_no_courses_filter=false, int $a_course_group_filter=0, int $a_role_filter=0, ?array $a_user_folder_filter=null, ?array $a_additional_fields=null, ?array $a_user_filter=null, string $a_first_letter="", string $a_authentication_filter="")
Get data for user administration list.
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
const IL_CAL_DAY
setUserFolder(?array $a_fold_id)
Set user folder filter reference id of user folder or category (local user administration) ...
const DEFAULT_ORDER_FIELD
setUserFilter(array $a_filter)
Array with user ids to query against.
setOrderDirection(string $a_dir)
Set order direction &#39;asc&#39; or &#39;desc&#39; Default is &#39;asc&#39;.
setCourseGroupFilter(int $a_cg_id)
Set course / group filter object_id of course or group.
global $DIC
Definition: shib_login.php:22
const UDF_TYPE_TEXT
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
setActionFilter(string $a_activation)
Set activation filter &#39;active&#39; or &#39;inactive&#39; or empty.
const UDF_TYPE_WYSIWYG
array $additional_fields
setUdfFilter(array $a_val)
Set udf filter.
$id
plugin.php for ilComponentBuildPluginInfoObjectiveTest::testAddPlugins
Definition: plugin.php:23
setAccessFilter(bool $a_access)
set filter for user that are limited but has access
static _lookupType(int $id, bool $reference=false)
setAuthenticationFilter(string $a_authentication)
Set authentication filter.
setNoGroupFilter(bool $a_no_group)