ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilObjUser.php
Go to the documentation of this file.
1 <?php
2 
24 use ILIAS\Data\DateFormat\Factory as DateFormatFactory;
27 
34 class ilObjUser extends ilObject
35 {
36  public const NO_AVATAR_RID = '-';
37  public const PASSWD_PLAIN = 'plain';
38  public const PASSWD_CRYPTED = 'crypted';
39  protected string $ext_account = '';
40  protected string $time_limit_message = '';
41  protected bool $time_limit_unlimited = false;
42  protected ?int $time_limit_until = null;
43  protected ?int $time_limit_from = null;
44  protected ?int $time_limit_owner = null;
45  protected string $last_login = '';
46 
47  public string $login = '';
48  protected string $passwd = ''; // password encoded in the format specified by $passwd_type
49  protected string $passwd_type = '';
50  // specifies the password format.
51  // value: ilObjUser::PASSWD_PLAIN or ilObjUser::PASSWD_CRYPTED.
52  // Differences between password format in class ilObjUser and
53  // in table usr_data:
54  // Class ilObjUser supports two different password types
55  // (plain and crypted) and it uses the variables $passwd
56  // and $passwd_type to store them.
57  // Table usr_data supports only two different password types
58  // (md5 and bcrypt) and it uses the columns 'passwd' and 'passwd_type' to store them.
59  // The conversion between these two storage layouts is done
60  // in the methods that perform SQL statements. All other
61  // methods work exclusively with the $passwd and $passwd_type
62  // variables.
63  protected ?string $password_encoding_type = null; // The encoding algorithm of the user's password stored in the database
64  // A salt used to encrypt the user's password
65  protected ?string $password_salt = null;
66  public string $gender = ''; // 'm' or 'f'
67  public string $utitle = ''; // user title (keep in mind, that we derive $title from object also!)
68  public string $firstname = '';
69  public string $lastname = '';
70  protected ?string $birthday = null;
71  public string $fullname = ''; // title + firstname + lastname in one string
72  public string $institution = '';
73  public string $department = '';
74  public string $street = '';
75  public string $city = '';
76  public string $zipcode = '';
77  public string $country = '';
78  public string $sel_country = '';
79  public string $phone_office = '';
80  public string $phone_home = '';
81  public string $phone_mobile = '';
82  public string $fax = '';
83  public string $email = '';
84  protected ?string $second_email = null;
85  public string $hobby = '';
86  public string $matriculation = '';
87  public string $referral_comment = '';
88  public ?string $approve_date = null;
89  public ?string $agree_date = null;
90  public int $active = 0;
91  public string $client_ip = ''; // client ip to check before login
92  public ?string $auth_mode = null; // authentication mode
93  public ?string $latitude = null;
94  public ?string $longitude = null;
95  public ?string $loc_zoom = null;
96  public int $last_password_change_ts = 0;
97  protected bool $passwd_policy_reset = false;
98  public int $login_attempts = 0;
99  public array $user_defined_data = []; // Missing array type.
101  protected array $oldPrefs = [];
103  public array $prefs = [];
104  public string $skin = '';
105  protected static array $personal_image_cache = [];
106  protected ?string $inactivation_date = null;
107  private bool $is_self_registered = false; // flag for self registered users
108  protected string $org_units = ''; // ids of assigned org-units, comma seperated
110  protected array $interests_general = [];
112  protected array $interests_help_offered = [];
114  protected array $interests_help_looking = [];
115  protected string $last_profile_prompt = ''; // timestamp
116  protected string $first_login = ''; // timestamp
117  protected bool $profile_incomplete = false;
118  protected ?string $avatar_rid = null;
119 
120  protected DateFormatFactory $date_format_factory;
122  private Services $irss;
123 
124  public function __construct(
125  int $a_user_id = 0,
126  bool $a_call_by_reference = false
127  ) {
128  global $DIC;
129 
130  $this->type = 'usr';
131  parent::__construct($a_user_id, $a_call_by_reference);
132 
133  $this->cron_delete_user_reminder_mail = new ilCronDeleteInactiveUserReminderMail($this->db);
134  $this->irss = $DIC->resourceStorage();
135  $this->auth_mode = 'default';
136  $this->passwd_type = self::PASSWD_PLAIN;
137  if ($a_user_id > 0) {
138  $this->setId($a_user_id);
139  $this->read();
140  } else {
141  $this->prefs = [];
142  $this->prefs['language'] = $this->ilias->ini->readVariable('language', 'default');
143  $this->skin = $this->ilias->ini->readVariable('layout', 'skin');
144  $this->prefs['skin'] = $this->skin;
145  $this->prefs['style'] = $this->ilias->ini->readVariable('layout', 'style');
146  }
147 
148  $this->app_event_handler = $DIC['ilAppEventHandler'];
149  $this->date_format_factory = (new DataFactory())->dateFormat();
150  }
151 
157  public function read(): void
158  {
159  global $DIC;
160 
161  $ilErr = $DIC['ilErr'];
162  $ilDB = $this->db;
163 
164  $r = $ilDB->queryF('SELECT * FROM usr_data ' .
165  'WHERE usr_id= %s', ['integer'], [$this->id]);
166 
167  if ($data = $ilDB->fetchAssoc($r)) {
168  // convert password storage layout used by table usr_data into
169  // storage layout used by class ilObjUser
170  $data['passwd_type'] = self::PASSWD_CRYPTED;
171 
172  // this assign must not be set via $this->assignData($data)
173  // because this method will be called on profile updates and
174  // would set this values to 0, because they arent posted from form
175  $this->setLastPasswordChangeTS($data['last_password_change']);
176  $this->setLoginAttempts($data['login_attempts']);
177  $this->setPasswordPolicyResetStatus((bool) $data['passwd_policy_reset']);
178 
179  // fill member vars in one shot
180  $this->assignData($data);
181 
182  //get userpreferences from usr_pref table
183  $this->readPrefs();
184 
185  if (!isset($this->prefs['language']) || $this->prefs['language'] === '') {
186  $this->prefs['language'] = $this->oldPrefs['language'] ?? '';
187  }
188 
189  if (
190  !isset($this->prefs['skin']) || $this->prefs['skin'] === '' ||
191  !ilStyleDefinition::skinExists($this->prefs['skin'])
192  ) {
193  $this->prefs['skin'] = $this->oldPrefs['skin'] ?? '';
194  }
195 
196  $this->skin = $this->prefs['skin'];
197 
198  if (
199  !isset($this->prefs['style']) ||
200  $this->prefs['style'] === '' ||
201  !ilStyleDefinition::styleExists($this->prefs['style']) ||
202  (
203  !ilStyleDefinition::skinExists($this->skin) &&
204  ilStyleDefinition::styleExistsForSkinId($this->skin, $this->prefs['style'])
205  )
206  ) {
207  //load default (css)
208  $this->prefs['skin'] = $this->ilias->ini->readVariable('layout', 'skin');
209  $this->prefs['style'] = $this->ilias->ini->readVariable('layout', 'style');
210  }
211  } else {
212  $ilErr->raiseError('<b>Error: There is no dataset with id ' .
213  $this->id . '!</b><br />class: ' . get_class($this) . '<br />Script: ' . __FILE__ .
214  '<br />Line: ' . __LINE__, $ilErr->FATAL);
215  }
216 
217  $this->readMultiTextFields();
218  $this->readUserDefinedFields();
219 
220  parent::read();
221  }
222 
223  public function getPasswordEncodingType(): ?string
224  {
226  }
227 
228  public function setPasswordEncodingType(?string $password_encryption_type): void
229  {
230  $this->password_encoding_type = $password_encryption_type;
231  }
232 
233  public function getPasswordSalt(): ?string
234  {
235  return $this->password_salt;
236  }
237 
238  public function setPasswordSalt(?string $password_salt): void
239  {
240  $this->password_salt = $password_salt;
241  }
242 
247  public function assignData(array $a_data): void
248  {
249  global $DIC;
250 
251  $ilErr = $DIC['ilErr'];
252 
253  // basic personal data
254  $this->setLogin($a_data['login'] ?? '');
255  if (!($a_data['passwd_type'] ?? false)) {
256  $ilErr->raiseError('<b>Error: passwd_type missing in function assignData(). ' .
257  $this->id . '!</b><br />class: ' . get_class($this) . '<br />Script: '
258  . __FILE__ . '<br />Line: ' . __LINE__, $ilErr->FATAL);
259  }
260  if (($a_data['passwd'] ?? '') != '********' && strlen($a_data['passwd'] ?? '')) {
261  $this->setPasswd($a_data['passwd'] ?? '', $a_data['passwd_type'] ?? '');
262  }
263 
264  $this->setGender((string) ($a_data['gender'] ?? ''));
265  $this->setUTitle((string) ($a_data['title'] ?? ''));
266  $this->setFirstname((string) ($a_data['firstname'] ?? ''));
267  $this->setLastname((string) ($a_data['lastname'] ?? ''));
268  $this->setFullname();
269  if (isset($a_data['birthday']) && is_string($a_data['birthday'])) {
270  $this->setBirthday($a_data['birthday']);
271  } else {
272  $this->setBirthday(null);
273  }
274 
275  // address data
276  $this->setInstitution((string) ($a_data['institution'] ?? ''));
277  $this->setDepartment((string) ($a_data['department'] ?? ''));
278  $this->setStreet((string) ($a_data['street'] ?? ''));
279  $this->setCity((string) ($a_data['city'] ?? ''));
280  $this->setZipcode((string) ($a_data['zipcode'] ?? ''));
281  $this->setCountry((string) ($a_data['country'] ?? ''));
282  $this->setSelectedCountry((string) ($a_data['sel_country'] ?? ''));
283  $this->setPhoneOffice((string) ($a_data['phone_office'] ?? ''));
284  $this->setPhoneHome((string) ($a_data['phone_home'] ?? ''));
285  $this->setPhoneMobile((string) ($a_data['phone_mobile'] ?? ''));
286  $this->setFax((string) ($a_data['fax'] ?? ''));
287  $this->setMatriculation((string) ($a_data['matriculation'] ?? ''));
288  $this->setEmail((string) ($a_data['email'] ?? ''));
289  $this->setSecondEmail((string) ($a_data['second_email'] ?? null));
290  $this->setHobby((string) ($a_data['hobby'] ?? ''));
291  $this->setClientIP((string) ($a_data['client_ip'] ?? ''));
292  $this->setPasswordEncodingType($a_data['passwd_enc_type'] ?? null);
293  $this->setPasswordSalt($a_data['passwd_salt'] ?? null);
294 
295  // other data
296  $this->setLatitude($a_data['latitude'] ?? null);
297  $this->setLongitude($a_data['longitude'] ?? null);
298  $this->setLocationZoom($a_data['loc_zoom'] ?? null);
299 
300  // system data
301  $this->setLastLogin((string) ($a_data['last_login'] ?? ''));
302  $this->setFirstLogin((string) ($a_data['first_login'] ?? ''));
303  $this->setLastProfilePrompt((string) ($a_data['last_profile_prompt'] ?? ''));
304  $this->setLastUpdate((string) ($a_data['last_update'] ?? ''));
305  $this->create_date = $a_data['create_date'] ?? '';
306  $this->setComment((string) ($a_data['referral_comment'] ?? ''));
307  $this->approve_date = ($a_data['approve_date'] ?? null);
308  $this->active = ($a_data['active'] ?? 0);
309  $this->agree_date = ($a_data['agree_date'] ?? null);
310 
311  $this->setInactivationDate((string) ($a_data['inactivation_date'] ?? null));
312 
313  // time limitation
314  $this->setTimeLimitOwner((int) ($a_data['time_limit_owner'] ?? 0));
315  $this->setTimeLimitUnlimited((bool) ($a_data['time_limit_unlimited'] ?? false));
316  $this->setTimeLimitFrom((int) ($a_data['time_limit_from'] ?? 0));
317  $this->setTimeLimitUntil((int) ($a_data['time_limit_until'] ?? 0));
318  $this->setTimeLimitMessage((string) ($a_data['time_limit_message'] ?? ''));
319 
320  // user profile incomplete?
321  $this->setProfileIncomplete((bool) ($a_data['profile_incomplete'] ?? false));
322 
323  //authentication
324  $this->setAuthMode((string) ($a_data['auth_mode'] ?? null));
325  $this->setExternalAccount((string) ($a_data['ext_account'] ?? ''));
326 
327  $this->setIsSelfRegistered((bool) ($a_data['is_self_registered'] ?? false));
328 
329  // Avatar
330  $this->avatar_rid = (string) ($a_data['rid'] ?? self::NO_AVATAR_RID);
331  }
332 
338  public function saveAsNew(): void
339  {
340  global $DIC;
341 
342  $ilAppEventHandler = $DIC['ilAppEventHandler'];
343 
344  $ilErr = $DIC['ilErr'];
345  $ilDB = $this->db;
346  $pw_value = '';
347 
348  switch ($this->passwd_type) {
349  case self::PASSWD_PLAIN:
350  if (strlen($this->passwd)) {
351  LocalUserPasswordManager::getInstance()->encodePassword($this, $this->passwd);
352  $pw_value = $this->getPasswd();
353  } else {
354  $pw_value = $this->passwd;
355  }
356  break;
357 
358  case self::PASSWD_CRYPTED:
359  $pw_value = $this->passwd;
360  break;
361 
362  default:
363  $ilErr->raiseError('<b>Error: passwd_type missing in function saveAsNew. ' .
364  $this->id . '!</b><br />class: ' . get_class($this) . '<br />Script: ' . __FILE__ .
365  '<br />Line: ' . __LINE__, $ilErr->FATAL);
366  }
367 
368  if (!$this->active) {
370  } else {
371  $this->setInactivationDate(null);
372  }
373 
374  $insert_array = [
375  'usr_id' => ['integer', $this->id],
376  'login' => ['text', $this->login],
377  'passwd' => ['text', $pw_value],
378  'passwd_enc_type' => ['text', $this->getPasswordEncodingType()],
379  'passwd_salt' => ['text', $this->getPasswordSalt()],
380  'firstname' => ['text', $this->firstname],
381  'lastname' => ['text', $this->lastname],
382  'title' => ['text', $this->utitle],
383  'gender' => ['text', $this->gender],
384  'email' => ['text', trim($this->email ?? '')],
385  'second_email' => ['text', trim($this->second_email ?? '')],
386  'hobby' => ['text', $this->hobby],
387  'institution' => ['text', $this->institution],
388  'department' => ['text', $this->department],
389  'street' => ['text', $this->street],
390  'city' => ['text', $this->city],
391  'zipcode' => ['text', $this->zipcode],
392  'country' => ['text', $this->country],
393  'sel_country' => ['text', $this->sel_country],
394  'phone_office' => ['text', $this->phone_office],
395  'phone_home' => ['text', $this->phone_home],
396  'phone_mobile' => ['text', $this->phone_mobile],
397  'fax' => ['text', $this->fax],
398  'birthday' => ['date', $this->getBirthday()],
399  'last_login' => ['timestamp', null],
400  'first_login' => ['timestamp', null],
401  'last_profile_prompt' => ['timestamp', null],
402  'last_update' => ['timestamp', ilUtil::now()],
403  'create_date' => ['timestamp', ilUtil::now()],
404  'referral_comment' => ['text', $this->referral_comment],
405  'matriculation' => ['text', $this->matriculation],
406  'client_ip' => ['text', $this->client_ip],
407  'approve_date' => ['timestamp', $this->approve_date],
408  'agree_date' => ['timestamp', $this->agree_date],
409  'active' => ['integer', $this->active],
410  'time_limit_unlimited' => ['integer', $this->getTimeLimitUnlimited()],
411  'time_limit_until' => ['integer', $this->getTimeLimitUntil()],
412  'time_limit_from' => ['integer', $this->getTimeLimitFrom()],
413  'time_limit_owner' => ['integer', $this->getTimeLimitOwner()],
414  'auth_mode' => ['text', $this->getAuthMode()],
415  'ext_account' => ['text', $this->getExternalAccount()],
416  'profile_incomplete' => ['integer', $this->getProfileIncomplete()],
417  'latitude' => ['text', $this->latitude],
418  'longitude' => ['text', $this->longitude],
419  'loc_zoom' => ['integer', (int) $this->loc_zoom],
420  'last_password_change' => ['integer', $this->last_password_change_ts],
421  'passwd_policy_reset' => ['integer', (int) $this->passwd_policy_reset],
422  'inactivation_date' => ['timestamp', $this->inactivation_date],
423  'is_self_registered' => ['integer', (int) $this->is_self_registered],
424  ];
425  $ilDB->insert('usr_data', $insert_array);
426 
427  $this->updateMultiTextFields(true);
428  $this->updateUserDefinedFields();
429 
430  // CREATE ENTRIES FOR MAIL BOX
431  $mbox = new ilMailbox($this->id);
432  $mbox->createDefaultFolder();
433 
434  $mail_options = new ilMailOptions($this->id);
435  $mail_options->createMailOptionsEntry();
436 
437  $ilAppEventHandler->raise(
438  'components/ILIAS/User',
439  'afterCreate',
440  ['user_obj' => $this]
441  );
442  }
443 
444  public function update(): bool
445  {
446  global $DIC;
447 
448  $ilErr = $DIC['ilErr'];
449  $ilDB = $this->db;
450  $ilAppEventHandler = $this->app_event_handler;
451 
452  $this->syncActive();
453 
454  if ($this->getStoredActive($this->id) && !$this->active) {
456  } elseif ($this->active) {
457  $this->setInactivationDate(null);
458  }
459 
460  $update_array = [
461  'gender' => ['text', $this->gender],
462  'title' => ['text', $this->utitle],
463  'firstname' => ['text', substr($this->firstname, 0, 128)],
464  'lastname' => ['text', substr($this->lastname, 0, 128)],
465  'email' => ['text', substr(trim($this->email), 0, 128)],
466  'second_email' => ['text', trim($this->second_email ?? '')],
467  'birthday' => ['date', $this->getBirthday()],
468  'hobby' => ['text', $this->hobby],
469  'institution' => ['text', $this->institution],
470  'department' => ['text', $this->department],
471  'street' => ['text', $this->street],
472  'city' => ['text', $this->city],
473  'zipcode' => ['text', $this->zipcode],
474  'country' => ['text', $this->country],
475  'sel_country' => ['text', $this->sel_country],
476  'phone_office' => ['text', $this->phone_office],
477  'phone_home' => ['text', $this->phone_home],
478  'phone_mobile' => ['text', $this->phone_mobile],
479  'fax' => ['text', $this->fax],
480  'referral_comment' => ['text', $this->referral_comment],
481  'matriculation' => ['text', $this->matriculation],
482  'client_ip' => ['text', $this->client_ip],
483  'approve_date' => ['timestamp', $this->approve_date],
484  'active' => ['integer', $this->active],
485  'time_limit_unlimited' => ['integer', $this->getTimeLimitUnlimited()],
486  'time_limit_until' => ['integer', $this->getTimeLimitUntil()],
487  'time_limit_from' => ['integer', $this->getTimeLimitFrom()],
488  'time_limit_owner' => ['integer', $this->getTimeLimitOwner()],
489  'time_limit_message' => ['integer', $this->getTimeLimitMessage()],
490  'profile_incomplete' => ['integer', $this->getProfileIncomplete()],
491  'auth_mode' => ['text', $this->getAuthMode()],
492  'ext_account' => ['text', $this->getExternalAccount()],
493  'latitude' => ['text', $this->latitude],
494  'longitude' => ['text', $this->longitude],
495  'loc_zoom' => ['integer', (int) $this->loc_zoom],
496  'login_attempts' => ['integer', $this->login_attempts],
497  'last_password_change' => ['integer', $this->last_password_change_ts],
498  'passwd_policy_reset' => ['integer', $this->passwd_policy_reset],
499  'last_update' => ['timestamp', ilUtil::now()],
500  'inactivation_date' => ['timestamp', $this->inactivation_date],
501  'reg_hash' => ['text', null],
502  'rid' => [
503  'text',
504  ($this->avatar_rid ?? self::NO_AVATAR_RID)
505  ],
506  ];
507 
508  if ($this->agree_date === null || (is_string($this->agree_date) && strtotime($this->agree_date) !== false)) {
509  $update_array['agree_date'] = ['timestamp', $this->agree_date];
510  }
511  switch ($this->passwd_type) {
512  case self::PASSWD_PLAIN:
513  if (strlen($this->passwd)) {
514  LocalUserPasswordManager::getInstance()->encodePassword($this, $this->passwd);
515  $update_array['passwd'] = ['text', $this->getPasswd()];
516  } else {
517  $update_array['passwd'] = ['text', $this->passwd];
518  }
519  break;
520 
521  case self::PASSWD_CRYPTED:
522  $update_array['passwd'] = ['text', $this->passwd];
523  break;
524 
525  default:
526  $ilErr->raiseError('<b>Error: passwd_type missing in function update()' . $this->id . '!</b><br />class: ' .
527  get_class($this) . '<br />Script: ' . __FILE__ . '<br />Line: ' . __LINE__, $ilErr->FATAL);
528  }
529 
530  $update_array['passwd_enc_type'] = ['text', $this->getPasswordEncodingType()];
531  $update_array['passwd_salt'] = ['text', $this->getPasswordSalt()];
532 
533  $ilDB->update('usr_data', $update_array, ['usr_id' => ['integer', $this->id]]);
534 
535  $this->updateMultiTextFields();
536 
537  $this->writePrefs();
538 
539  // update user defined fields
540  $this->updateUserDefinedFields();
541 
542  parent::update();
543  $this->updateOwner();
544 
545  $this->read();
546 
547  $ilAppEventHandler->raise(
548  'components/ILIAS/User',
549  'afterUpdate',
550  ['user_obj' => $this]
551  );
552 
553  return true;
554  }
555 
559  public function writeAccepted(): void
560  {
561  $ilDB = $this->db;
562  $ilDB->manipulateF('UPDATE usr_data SET agree_date = ' . $ilDB->now() .
563  ' WHERE usr_id = %s', ['integer'], [$this->getId()]);
564  }
565 
566  private static function _lookup(
567  int $a_user_id,
568  string $a_field
569  ): ?string {
570  global $DIC;
571 
572  $ilDB = $DIC->database();
573 
574  $res = $ilDB->queryF(
575  'SELECT ' . $a_field . ' FROM usr_data WHERE usr_id = %s',
576  ['integer'],
577  [$a_user_id]
578  );
579 
580  while ($set = $ilDB->fetchAssoc($res)) {
581  return $set[$a_field];
582  }
583  return null;
584  }
585 
586  public static function _lookupFullname(int $a_user_id): string
587  {
588  global $DIC;
589 
590  $fullname = '';
591  $ilDB = $DIC['ilDB'];
592 
593  $set = $ilDB->queryF(
594  'SELECT title, firstname, lastname FROM usr_data WHERE usr_id = %s',
595  ['integer'],
596  [$a_user_id]
597  );
598 
599  if ($rec = $ilDB->fetchAssoc($set)) {
600  if ($rec['title']) {
601  $fullname = $rec['title'] . ' ';
602  }
603  if ($rec['firstname']) {
604  $fullname .= $rec['firstname'] . ' ';
605  }
606  if ($rec['lastname']) {
607  $fullname .= $rec['lastname'];
608  }
609  }
610  return $fullname;
611  }
612 
613  public static function _lookupEmail(int $a_user_id): string
614  {
615  return self::_lookup($a_user_id, 'email') ?? '';
616  }
617 
618  public static function _lookupGender(int $a_user_id): string
619  {
620  return (string) self::_lookup($a_user_id, 'gender') ?? '';
621  }
622 
623  public static function _lookupClientIP(int $a_user_id): string
624  {
625  return self::_lookup($a_user_id, 'client_ip') ?? '';
626  }
627 
632  public static function _lookupName(int $a_user_id): array
633  {
634  global $DIC;
635 
636  $ilDB = $DIC['ilDB'];
637 
638  $res = $ilDB->queryF(
639  'SELECT firstname, lastname, title, login FROM usr_data WHERE usr_id = %s',
640  ['integer'],
641  [$a_user_id]
642  );
643  if ($user_rec = $ilDB->fetchAssoc($res)) {
644  return ['user_id' => $a_user_id,
645  'firstname' => $user_rec['firstname'],
646  'lastname' => $user_rec['lastname'],
647  'title' => $user_rec['title'],
648  'login' => $user_rec['login']
649  ];
650  }
651  return ['user_id' => 0,
652  'firstname' => '',
653  'lastname' => '',
654  'title' => '',
655  'login' => ''
656  ];
657  }
658 
663  public static function _lookupFields(int $a_user_id): array // Missing array type.
664  {
665  global $DIC;
666 
667  $ilDB = $DIC['ilDB'];
668 
669  $res = $ilDB->queryF(
670  'SELECT * FROM usr_data WHERE usr_id = %s',
671  ['integer'],
672  [$a_user_id]
673  );
674  $user_rec = $ilDB->fetchAssoc($res);
675  return $user_rec;
676  }
677 
678  public static function _lookupLogin(int $a_user_id): string
679  {
680  return (string) self::_lookup($a_user_id, 'login') ?? '';
681  }
682 
683  public static function _lookupExternalAccount(int $a_user_id): string
684  {
685  return (string) self::_lookup($a_user_id, 'ext_account') ?? '';
686  }
687 
692  public static function _lookupId($a_user_str)
693  {
694  global $DIC;
695 
696  $ilDB = $DIC['ilDB'];
697 
698  if (!is_array($a_user_str)) {
699  $res = $ilDB->queryF(
700  'SELECT usr_id FROM usr_data WHERE login = %s',
701  ['text'],
702  [$a_user_str]
703  );
704 
705  $user_rec = $ilDB->fetchAssoc($res);
706  if (is_array($user_rec)) {
707  return (int) $user_rec['usr_id'];
708  }
709 
710  return null;
711  }
712 
713  $set = $ilDB->query(
714  'SELECT usr_id FROM usr_data ' .
715  ' WHERE ' . $ilDB->in('login', $a_user_str, false, 'text')
716  );
717 
718  $ids = [];
719  while ($rec = $ilDB->fetchAssoc($set)) {
720  $ids[] = (int) $rec['usr_id'];
721  }
722 
723  return $ids;
724  }
725 
726  public static function _lookupLastLogin(int $a_user_id): string
727  {
728  return self::_lookup($a_user_id, 'last_login') ?? '';
729  }
730 
731  public static function _lookupFirstLogin(int $a_user_id): string
732  {
733  return self::_lookup($a_user_id, 'first_login') ?? '';
734  }
735 
736 
741  public function refreshLogin(): void
742  {
743  $ilDB = $this->db;
744 
745  $ilDB->manipulateF(
746  'UPDATE usr_data SET ' .
747  'last_login = ' . $ilDB->now() .
748  ' WHERE usr_id = %s',
749  ['integer'],
750  [$this->id]
751  );
752 
753  if ($this->getFirstLogin() == '') {
754  $ilDB->manipulateF(
755  'UPDATE usr_data SET ' .
756  'first_login = ' . $ilDB->now() .
757  ' WHERE usr_id = %s',
758  ['integer'],
759  [$this->id]
760  );
761  $this->app_event_handler->raise(
762  'components/ILIAS/User',
763  'firstLogin',
764  ['user_obj' => $this]
765  );
766  }
767  }
768 
769 
778  public function resetPassword(
779  string $raw,
780  string $raw_retype
781  ): bool {
782  $ilDB = $this->db;
783 
784  if (func_num_args() != 2) {
785  return false;
786  }
787 
788  if (!isset($raw) || !isset($raw_retype)) {
789  return false;
790  }
791 
792  if ($raw != $raw_retype) {
793  return false;
794  }
795 
796  LocalUserPasswordManager::getInstance()->encodePassword($this, $raw);
797 
798  $ilDB->manipulateF(
799  'UPDATE usr_data
800  SET passwd = %s, passwd_enc_type = %s, passwd_salt = %s
801  WHERE usr_id = %s',
802  ['text', 'text', 'text', 'integer'],
803  [$this->getPasswd(), $this->getPasswordEncodingType(), $this->getPasswordSalt(), $this->getId()]
804  );
805 
806  return true;
807  }
808 
812  public static function _doesLoginnameExistInHistory(string $a_login): bool
813  {
814  global $DIC;
815 
816  $ilDB = $DIC->database();
817 
818  $res = $ilDB->queryF(
819  '
820  SELECT * FROM loginname_history
821  WHERE login = %s',
822  ['text'],
823  [$a_login]
824  );
825 
826  return (bool) $ilDB->fetchAssoc($res);
827  }
828 
835  public static function _getLastHistoryDataByUserId(int $a_usr_id): array
836  {
837  global $DIC;
838 
839  $ilDB = $DIC['ilDB'];
840 
841  $ilDB->setLimit(1, 0);
842  $res = $ilDB->queryF(
843  '
844  SELECT login, history_date FROM loginname_history
845  WHERE usr_id = %s ORDER BY history_date DESC',
846  ['integer'],
847  [$a_usr_id]
848  );
849  $row = $ilDB->fetchAssoc($res);
850  if (!is_array($row) || !count($row)) {
851  throw new ilUserException('');
852  }
853 
854  return [
855  $row['login'], $row['history_date']
856  ];
857  }
858 
866  public function updateLogin(string $a_login): bool
867  {
868  global $DIC;
869 
870  $ilDB = $this->db;
871  $ilSetting = $DIC['ilSetting'];
872  $lng = $DIC->language();
873 
874  if (func_num_args() != 1) {
875  return false;
876  }
877 
878  if (!isset($a_login)) {
879  return false;
880  }
881 
882  $former_login = self::_lookupLogin($this->getId());
883 
884  // Update not necessary
885  if (0 == strcmp($a_login, $former_login)) {
886  return false;
887  }
888 
889  try {
890  $last_history_entry = self::_getLastHistoryDataByUserId($this->getId());
891  } catch (ilUserException $e) {
892  $last_history_entry = null;
893  }
894 
895  // throw exception if the desired loginame is already in history and it is not allowed to reuse it
896  if ((int) $ilSetting->get('allow_change_loginname') &&
897  (int) $ilSetting->get('reuse_of_loginnames') == 0 &&
898  self::_doesLoginnameExistInHistory($a_login)) {
899  throw new ilUserException($lng->txt('loginname_already_exists'));
900  } elseif ((int) $ilSetting->get('allow_change_loginname') &&
901  (int) $ilSetting->get('loginname_change_blocking_time') &&
902  is_array($last_history_entry) &&
903  $last_history_entry[1] + (int) $ilSetting->get('loginname_change_blocking_time') > time()) {
904  throw new ilUserException(
905  sprintf(
906  $lng->txt('changing_loginname_not_possible_info'),
908  new ilDateTime($last_history_entry[1], IL_CAL_UNIX)
909  ),
911  new ilDateTime(($last_history_entry[1] + (int) $ilSetting->get('loginname_change_blocking_time')), IL_CAL_UNIX)
912  )
913  )
914  );
915  } else {
916  // log old loginname in history
917  if ((int) $ilSetting->get('allow_change_loginname') &&
918  (int) $ilSetting->get('create_history_loginname')) {
919  self::_writeHistory($this->getId(), $former_login);
920  }
921 
922  //update login
923  $this->login = $a_login;
924 
925  $ilDB->manipulateF(
926  '
927  UPDATE usr_data
928  SET login = %s
929  WHERE usr_id = %s',
930  ['text', 'integer'],
931  [$this->getLogin(), $this->getId()]
932  );
933  }
934 
935  return true;
936  }
937 
938  public function writePref(
939  string $a_keyword,
940  string $a_value
941  ): void {
942  self::_writePref($this->id, $a_keyword, $a_value);
943  $this->setPref($a_keyword, $a_value);
944  }
945 
946  public function deletePref(string $a_keyword): void
947  {
948  self::_deletePref($this->getId(), $a_keyword);
949  }
950 
951  public static function _deletePref(int $a_user_id, string $a_keyword): void
952  {
953  global $DIC;
954 
955  $ilDB = $DIC->database();
956 
957  $ilDB->manipulateF(
958  'DELETE FROM usr_pref WHERE usr_id = %s AND keyword = %s',
959  ['integer', 'text'],
960  [$a_user_id, $a_keyword]
961  );
962  }
963 
967  public static function _deleteAllPref(int $a_user_id): void
968  {
969  global $DIC;
970 
971  $ilDB = $DIC->database();
972 
973  $ilDB->manipulateF(
974  'DELETE FROM usr_pref WHERE usr_id = %s',
975  ['integer'],
976  [$a_user_id]
977  );
978  }
979 
980  public static function _writePref(
981  int $a_usr_id,
982  string $a_keyword,
983  string $a_value
984  ): void {
985  global $DIC;
986 
987  $ilDB = $DIC->database();
988  $ilDB->replace(
989  'usr_pref',
990  [
991  'usr_id' => ['integer', $a_usr_id],
992  'keyword' => ['text', $a_keyword],
993  ],
994  [
995  'value' => ['text',$a_value]
996  ]
997  );
998  }
999 
1000  public function writePrefs(): void
1001  {
1002  self::_deleteAllPref($this->id);
1003  foreach ($this->prefs as $keyword => $value) {
1004  self::_writePref($this->id, $keyword, (string) $value);
1005  }
1006  }
1007 
1008  public function getTimeZone(): string
1009  {
1010  if ($tz = $this->getPref('user_tz')) {
1011  return $tz;
1012  } else {
1013  $settings = ilCalendarSettings::_getInstance();
1014  return $settings->getDefaultTimeZone();
1015  }
1016  }
1017 
1018  public function getTimeFormat(): string
1019  {
1020  if ($format = $this->getPref('time_format')) {
1021  return $format;
1022  } else {
1023  $settings = ilCalendarSettings::_getInstance();
1024  return $settings->getDefaultTimeFormat();
1025  }
1026  }
1027 
1028  public function getDateFormat(): DateFormat
1029  {
1030  if (!($format = $this->getPref('date_format'))) {
1031  $settings = ilCalendarSettings::_getInstance();
1032  $format = $settings->getDefaultDateFormat();
1033  }
1034 
1035  switch ($format) {
1037  return $this->date_format_factory->germanShort();
1038 
1040  return $this->date_format_factory->americanShort();
1041 
1043  default:
1044  return $this->date_format_factory->standard();
1045  }
1046  }
1047 
1048  public function getDateTimeFormat(): DateFormat
1049  {
1051  return $this->date_format_factory->withTime24($this->getDateFormat());
1052  }
1053  return $this->date_format_factory->withTime12($this->getDateFormat());
1054  }
1055 
1056  public function setPref(string $a_keyword, ?string $a_value): void
1057  {
1058  if ($a_keyword != '') {
1059  $this->prefs[$a_keyword] = $a_value;
1060  }
1061  }
1062 
1063  public function getPref(string $a_keyword): ?string
1064  {
1065  return $this->prefs[$a_keyword] ?? null;
1066  }
1067 
1068  public function existsPref(string $a_keyword): bool
1069  {
1070  return (array_key_exists($a_keyword, $this->prefs));
1071  }
1072 
1073  public static function _lookupPref(
1074  int $a_usr_id,
1075  string $a_keyword
1076  ): ?string {
1077  global $DIC;
1078 
1079  $ilDB = $DIC['ilDB'];
1080 
1081  $query = 'SELECT * FROM usr_pref WHERE usr_id = ' . $ilDB->quote($a_usr_id, 'integer') . ' ' .
1082  'AND keyword = ' . $ilDB->quote($a_keyword, 'text');
1083  $res = $ilDB->query($query);
1084 
1085  while ($row = $res->fetchRow(ilDBConstants::FETCHMODE_OBJECT)) {
1086  return $row->value;
1087  }
1088  return null;
1089  }
1090 
1091  public function readPrefs(): void
1092  {
1093  if (is_array($this->prefs)) {
1094  $this->oldPrefs = $this->prefs;
1095  }
1096  $this->prefs = self::_getPreferences($this->id);
1097  }
1098 
1099  public function delete(): bool
1100  {
1101  global $DIC;
1102 
1103  $rbacadmin = $DIC->rbac()->admin();
1104  $ilDB = $this->db;
1105 
1106  // deassign from ldap groups
1108  $mapping->deleteUser($this->getId());
1109 
1110  // remove mailbox / update sent mails
1111  $mailbox = new ilMailbox($this->getId());
1112  $mailbox->delete();
1113  $mailbox->updateMailsOfDeletedUser($this->getLogin());
1114 
1115  // delete block settings
1117 
1118  // delete user_account
1119  $ilDB->manipulateF(
1120  'DELETE FROM usr_data WHERE usr_id = %s',
1121  ['integer'],
1122  [$this->getId()]
1123  );
1124 
1125  $this->deleteMultiTextFields();
1126 
1127  // delete user_prefs
1128  self::_deleteAllPref($this->getId());
1129 
1130  $this->removeUserPicture(false); // #8597
1131 
1132  // delete user_session
1134 
1135  // remove user from rbac
1136  $rbacadmin->removeUser($this->getId());
1137 
1138  // remove bookmarks
1139  // TODO: move this to class.ilBookmarkFolder
1140  $q = 'DELETE FROM bookmark_tree WHERE tree = ' .
1141  $ilDB->quote($this->getId(), 'integer');
1142  $ilDB->manipulate($q);
1143 
1144  $q = 'DELETE FROM bookmark_data WHERE user_id = ' .
1145  $ilDB->quote($this->getId(), 'integer');
1146  $ilDB->manipulate($q);
1147 
1148  // Delete crs entries
1149  ilObjCourse::_deleteUser($this->getId());
1150 
1151  // Delete user tracking
1153 
1155 
1156  // Delete Tracking data SCORM 2004 RTE
1158 
1159  // Delete Tracking data SCORM 1.2 RTE
1161 
1162  // remove all notifications
1164 
1165  // remove portfolios
1167 
1168  // remove workspace
1169  $tree = new ilWorkspaceTree($this->getId());
1170  $tree->cascadingDelete();
1171 
1172  // remove reminder entries
1173  $this->cron_delete_user_reminder_mail->removeSingleUserFromTable($this->getId());
1174 
1175  // badges
1177 
1178  // remove org unit assignments
1179  $ilOrgUnitUserAssignmentQueries = ilOrgUnitUserAssignmentQueries::getInstance();
1180  $ilOrgUnitUserAssignmentQueries->deleteAllAssignmentsOfUser($this->getId());
1181 
1182  // Delete user defined field entries
1184 
1185  // Delete clipboard entries
1186  $this->clipboardDeleteAll();
1187 
1188  // Reset owner
1189  $this->resetOwner();
1190 
1191  // Trigger deleteUser Event
1192  global $DIC;
1193 
1194  $ilAppEventHandler = $DIC['ilAppEventHandler'];
1195  $ilAppEventHandler->raise(
1196  'components/ILIAS/User',
1197  'deleteUser',
1198  ['usr_id' => $this->getId()]
1199  );
1200 
1201  // delete object data
1202  parent::delete();
1203  return true;
1204  }
1205 
1211  public function setFullname(): void
1212  {
1213  $this->fullname = ($this->utitle != '')
1214  ? $this->utitle . ' '
1215  : '';
1216  $this->fullname .= $this->firstname . ' ';
1217  $this->fullname .= $this->lastname;
1218  }
1219 
1232  public function getFullname(int $a_max_strlen = 0): string
1233  {
1234  if (!$a_max_strlen) {
1235  return ilUtil::stripSlashes($this->fullname);
1236  }
1237 
1238  if (strlen($this->fullname) <= $a_max_strlen) {
1239  return ilUtil::stripSlashes($this->fullname);
1240  }
1241 
1242  if ((strlen($this->utitle) + strlen($this->lastname) + 4) <= $a_max_strlen) {
1243  return ilUtil::stripSlashes($this->utitle . ' ' . substr($this->firstname, 0, 1) . '. ' . $this->lastname);
1244  }
1245 
1246  if ((strlen($this->firstname) + strlen($this->lastname) + 1) <= $a_max_strlen) {
1247  return ilUtil::stripSlashes($this->firstname . ' ' . $this->lastname);
1248  }
1249 
1250  if ((strlen($this->lastname) + 3) <= $a_max_strlen) {
1251  return ilUtil::stripSlashes(substr($this->firstname, 0, 1) . '. ' . $this->lastname);
1252  }
1253 
1254  return ilUtil::stripSlashes(substr($this->lastname, 0, $a_max_strlen));
1255  }
1256 
1257  public function setLogin(string $a_str): void
1258  {
1259  $this->login = $a_str;
1260  }
1261 
1262  public function getLogin(): string
1263  {
1264  return $this->login;
1265  }
1266 
1267  public function setPasswd(
1268  string $a_str,
1269  string $a_type = ilObjUser::PASSWD_PLAIN
1270  ): void {
1271  $this->passwd = $a_str;
1272  $this->passwd_type = $a_type;
1273  }
1274 
1278  public function getPasswd(): string
1279  {
1280  return $this->passwd;
1281  }
1282 
1286  public function getPasswdType(): string
1287  {
1288  return $this->passwd_type;
1289  }
1290 
1291  public function setGender(string $a_str): void
1292  {
1293  $this->gender = substr($a_str, -1);
1294  }
1295 
1296  public function getGender(): string
1297  {
1298  return $this->gender;
1299  }
1300 
1306  public function setUTitle(string $a_str): void
1307  {
1308  $this->utitle = $a_str;
1309  }
1310 
1311  public function getUTitle(): string
1312  {
1313  return $this->utitle;
1314  }
1315 
1316  public function setFirstname(string $a_str): void
1317  {
1318  $this->firstname = $a_str;
1319  }
1320 
1321  public function getFirstname(): string
1322  {
1323  return $this->firstname;
1324  }
1325 
1326  public function setLastname(string $a_str): void
1327  {
1328  $this->lastname = $a_str;
1329  }
1330 
1331  public function getLastname(): string
1332  {
1333  return $this->lastname;
1334  }
1335 
1336  public function setInstitution(string $a_str): void
1337  {
1338  $this->institution = $a_str;
1339  }
1340 
1341  public function getInstitution(): string
1342  {
1343  return $this->institution;
1344  }
1345 
1346  public function setDepartment(string $a_str): void
1347  {
1348  $this->department = $a_str;
1349  }
1350 
1351  public function getDepartment(): string
1352  {
1353  return $this->department;
1354  }
1355 
1356  public function setStreet(string $a_str): void
1357  {
1358  $this->street = $a_str;
1359  }
1360 
1361  public function getStreet(): string
1362  {
1363  return $this->street;
1364  }
1365 
1366  public function setCity(string $a_str): void
1367  {
1368  $this->city = $a_str;
1369  }
1370 
1371  public function getCity(): string
1372  {
1373  return $this->city;
1374  }
1375 
1376  public function setZipcode(string $a_str): void
1377  {
1378  $this->zipcode = $a_str;
1379  }
1380 
1381  public function getZipcode(): string
1382  {
1383  return $this->zipcode;
1384  }
1385 
1386  public function setCountry(string $a_str): void
1387  {
1388  $this->country = $a_str;
1389  }
1390 
1391  public function getCountry(): string
1392  {
1393  return $this->country;
1394  }
1395 
1399  public function setSelectedCountry(string $a_val): void
1400  {
1401  $this->sel_country = $a_val;
1402  }
1403 
1407  public function getSelectedCountry(): string
1408  {
1409  return $this->sel_country;
1410  }
1411 
1412  public function setPhoneOffice(string $a_str): void
1413  {
1414  $this->phone_office = $a_str;
1415  }
1416 
1417  public function getPhoneOffice(): string
1418  {
1419  return $this->phone_office;
1420  }
1421 
1422  public function setPhoneHome(string $a_str): void
1423  {
1424  $this->phone_home = $a_str;
1425  }
1426 
1427  public function getPhoneHome(): string
1428  {
1429  return $this->phone_home;
1430  }
1431 
1432  public function setPhoneMobile(string $a_str): void
1433  {
1434  $this->phone_mobile = $a_str;
1435  }
1436 
1437  public function getPhoneMobile(): string
1438  {
1439  return $this->phone_mobile;
1440  }
1441 
1442  public function setFax(string $a_str): void
1443  {
1444  $this->fax = $a_str;
1445  }
1446 
1447  public function getFax(): string
1448  {
1449  return $this->fax;
1450  }
1451 
1452  public function setClientIP(string $a_str): void
1453  {
1454  $this->client_ip = $a_str;
1455  }
1456 
1457  public function getClientIP(): string
1458  {
1459  return $this->client_ip;
1460  }
1461 
1462  public function setMatriculation(string $a_str): void
1463  {
1464  $this->matriculation = $a_str;
1465  }
1466 
1467  public function getMatriculation(): string
1468  {
1469  return $this->matriculation;
1470  }
1471 
1472  public static function lookupMatriculation(int $a_usr_id): string
1473  {
1474  global $DIC;
1475 
1476  $ilDB = $DIC['ilDB'];
1477 
1478  $query = 'SELECT matriculation FROM usr_data ' .
1479  'WHERE usr_id = ' . $ilDB->quote($a_usr_id);
1480  $res = $ilDB->query($query);
1481  $row = $res->fetchRow(ilDBConstants::FETCHMODE_OBJECT);
1482  return $row->matriculation ?: '';
1483  }
1484 
1485  public function setEmail(string $a_str): void
1486  {
1487  $this->email = $a_str;
1488  }
1489 
1490  public function getEmail(): string
1491  {
1492  return $this->email;
1493  }
1494 
1495  public function getSecondEmail(): ?string
1496  {
1497  return $this->second_email;
1498  }
1499 
1500  public function setSecondEmail(?string $second_email): void
1501  {
1502  $this->second_email = $second_email;
1503  }
1504 
1505  public function setHobby(string $a_str): void
1506  {
1507  $this->hobby = $a_str;
1508  }
1509 
1510  public function getHobby(): string
1511  {
1512  return $this->hobby;
1513  }
1514 
1515  public function setLanguage(string $a_str): void
1516  {
1517  $this->setPref('language', $a_str);
1518  ilSession::clear('lang');
1519  }
1520 
1521  public function getLanguage(): string
1522  {
1523  return $this->prefs['language'];
1524  }
1525 
1526  public function setLastPasswordChangeTS(int $a_last_password_change_ts): void
1527  {
1528  $this->last_password_change_ts = $a_last_password_change_ts;
1529  }
1530 
1531  public function getLastPasswordChangeTS(): int
1532  {
1534  }
1535 
1536  public function getPasswordPolicyResetStatus(): bool
1537  {
1539  }
1540 
1541  public function setPasswordPolicyResetStatus(bool $status): void
1542  {
1543  $this->passwd_policy_reset = $status;
1544  }
1545 
1546  public static function _lookupLanguage(int $a_usr_id): string
1547  {
1548  global $DIC;
1549 
1550  $ilDB = $DIC->database();
1551  $lng = $DIC->language();
1552 
1553  $q = 'SELECT value FROM usr_pref WHERE usr_id= ' .
1554  $ilDB->quote($a_usr_id, 'integer') . ' AND keyword = ' .
1555  $ilDB->quote('language', 'text');
1556  $r = $ilDB->query($q);
1557 
1558  while ($row = $ilDB->fetchAssoc($r)) {
1559  return (string) $row['value'];
1560  }
1561  if (is_object($lng)) {
1562  return $lng->getDefaultLanguage();
1563  }
1564  return 'en';
1565  }
1566 
1567  public static function _writeExternalAccount(
1568  int $a_usr_id,
1569  string $a_ext_id
1570  ): void {
1571  global $DIC;
1572 
1573  $ilDB = $DIC['ilDB'];
1574 
1575  $ilDB->manipulateF(
1576  'UPDATE usr_data ' .
1577  ' SET ext_account = %s WHERE usr_id = %s',
1578  ['text', 'integer'],
1579  [$a_ext_id, $a_usr_id]
1580  );
1581  }
1582 
1583  public static function _writeAuthMode(int $a_usr_id, string $a_auth_mode): void
1584  {
1585  global $DIC;
1586 
1587  $ilDB = $DIC['ilDB'];
1588 
1589  $ilDB->manipulateF(
1590  'UPDATE usr_data ' .
1591  ' SET auth_mode = %s WHERE usr_id = %s',
1592  ['text', 'integer'],
1593  [$a_auth_mode, $a_usr_id]
1594  );
1595  }
1596 
1600  public function getCurrentLanguage(): string
1601  {
1602  return (string) ilSession::get('lang');
1603  }
1604 
1608  public function setCurrentLanguage(string $a_val): void
1609  {
1610  ilSession::set('lang', $a_val);
1611  }
1612 
1613  public function setLastLogin(string $a_str): void
1614  {
1615  $this->last_login = $a_str;
1616  }
1617 
1618  public function getLastLogin(): string
1619  {
1620  return $this->last_login;
1621  }
1622 
1623  public function setFirstLogin(string $a_str): void
1624  {
1625  $this->first_login = $a_str;
1626  }
1627 
1628  public function getFirstLogin(): string
1629  {
1630  return $this->first_login;
1631  }
1632 
1633  public function setLastProfilePrompt(string $a_str): void
1634  {
1635  $this->last_profile_prompt = $a_str;
1636  }
1637 
1638  public function getLastProfilePrompt(): string
1639  {
1641  }
1642 
1643  public function setLastUpdate(string $a_str): void
1644  {
1645  $this->last_update = $a_str;
1646  }
1647 
1648  public function getLastUpdate(): string
1649  {
1650  return $this->last_update;
1651  }
1652 
1653  public function setComment(string $a_str): void
1654  {
1655  $this->referral_comment = $a_str;
1656  }
1657 
1658  public function getComment(): string
1659  {
1660  return $this->referral_comment;
1661  }
1662 
1667  public function setApproveDate(?string $a_str): void
1668  {
1669  $this->approve_date = $a_str;
1670  }
1671 
1672  public function getApproveDate(): ?string
1673  {
1674  return $this->approve_date;
1675  }
1676 
1677  public function getAgreeDate(): ?string
1678  {
1679  return $this->agree_date;
1680  }
1681  public function setAgreeDate(?string $a_str): void
1682  {
1683  $this->agree_date = $a_str;
1684  }
1685 
1690  public function setActive(
1691  bool $a_active,
1692  int $a_owner = 0
1693  ): void {
1694  $this->setOwner($a_owner);
1695 
1696  if ($a_active) {
1697  $this->active = 1;
1698  $this->setApproveDate(date('Y-m-d H:i:s'));
1699  $this->setOwner($a_owner);
1700  } else {
1701  $this->active = 0;
1702  $this->setApproveDate(null);
1703  }
1704  }
1705 
1706  public function getActive(): bool
1707  {
1708  return (bool) $this->active;
1709  }
1710 
1711  public static function _lookupActive(int $a_usr_id): bool
1712  {
1713  global $DIC;
1714 
1715  $ilDB = $DIC['ilDB'];
1716 
1717  $query = 'SELECT usr_id FROM usr_data ' .
1718  'WHERE active = ' . $ilDB->quote(1, 'integer') . ' ' .
1719  'AND usr_id = ' . $ilDB->quote($a_usr_id, 'integer');
1720  $res = $ilDB->query($query);
1721  while ($res->fetchRow(ilDBConstants::FETCHMODE_OBJECT)) {
1722  return true;
1723  }
1724  return false;
1725  }
1726 
1732  public function syncActive(): void
1733  {
1734  $storedActive = 0;
1735  if ($this->getStoredActive($this->id)) {
1736  $storedActive = 1;
1737  }
1738 
1739  $currentActive = 0;
1740  if ($this->active) {
1741  $currentActive = 1;
1742  }
1743 
1744  if ((!empty($storedActive) && empty($currentActive)) ||
1745  (empty($storedActive) && !empty($currentActive))) {
1746  $this->setActive($currentActive, self::getUserIdByLogin(self::getLoginFromAuth()));
1747  }
1748  }
1749 
1753  public function getStoredActive(int $a_id): bool
1754  {
1755  return (bool) self::_lookup($a_id, 'active');
1756  }
1757 
1758  public function setSkin(string $a_str): void
1759  {
1760  $this->skin = $a_str;
1761  }
1762 
1763  public function setTimeLimitOwner(int $a_owner): void
1764  {
1765  $this->time_limit_owner = $a_owner;
1766  }
1767 
1768  public function getTimeLimitOwner(): int
1769  {
1770  return $this->time_limit_owner ?: 7;
1771  }
1772 
1773  public function setTimeLimitFrom(?int $a_from): void
1774  {
1775  $this->time_limit_from = $a_from;
1776  }
1777 
1778  public function getTimeLimitFrom(): ?int
1779  {
1780  return $this->time_limit_from;
1781  }
1782 
1783  public function setTimeLimitUntil(?int $a_until): void
1784  {
1785  $this->time_limit_until = $a_until;
1786  }
1787 
1788  public function getTimeLimitUntil(): ?int
1789  {
1790  return $this->time_limit_until;
1791  }
1792 
1793  public function setTimeLimitUnlimited(bool $a_unlimited): void
1794  {
1795  $this->time_limit_unlimited = $a_unlimited;
1796  }
1797 
1798  public function getTimeLimitUnlimited(): bool
1799  {
1801  }
1802 
1803  public function setTimeLimitMessage(string $a_time_limit_message): void
1804  {
1805  $this->time_limit_message = $a_time_limit_message;
1806  }
1807 
1808  public function getTimeLimitMessage(): string
1809  {
1811  }
1812 
1813  public function setLoginAttempts(int $a_login_attempts): void
1814  {
1815  $this->login_attempts = $a_login_attempts;
1816  }
1817 
1818  public function getLoginAttempts(): int
1819  {
1820  return $this->login_attempts;
1821  }
1822 
1823  public function checkTimeLimit(): bool
1824  {
1825  if ($this->getTimeLimitUnlimited()) {
1826  return true;
1827  }
1828  if ($this->getTimeLimitFrom() < time() and $this->getTimeLimitUntil() > time()) {
1829  return true;
1830  }
1831  return false;
1832  }
1833 
1834  public function setProfileIncomplete(bool $a_prof_inc): void
1835  {
1836  $this->profile_incomplete = $a_prof_inc;
1837  }
1838 
1839  public function getProfileIncomplete(): bool
1840  {
1841  if ($this->id == ANONYMOUS_USER_ID) {
1842  return false;
1843  }
1845  }
1846 
1847  public function isPasswordChangeDemanded(): bool
1848  {
1849  if ($this->id == ANONYMOUS_USER_ID) {
1850  return false;
1851  }
1852 
1853  if ($this->id == SYSTEM_USER_ID) {
1854  if (
1855  LocalUserPasswordManager::getInstance()->verifyPassword($this, base64_decode('aG9tZXI=')) &&
1857  ) {
1858  return true;
1859  } else {
1860  return false;
1861  }
1862  }
1863 
1864  $security = ilSecuritySettings::_getInstance();
1865 
1866  $authModeAllowsPasswordChange = !ilAuthUtils::_needsExternalAccountByAuthMode($this->getAuthMode(true));
1867  $passwordResetOnFirstLogin = (
1868  $security->isPasswordChangeOnFirstLoginEnabled() &&
1869  $this->getLastPasswordChangeTS() == 0 && $this->is_self_registered == false
1870  );
1871  $passwordResetOnChangedPolicy = $this->getPasswordPolicyResetStatus();
1872 
1873  return ($authModeAllowsPasswordChange && ($passwordResetOnFirstLogin || $passwordResetOnChangedPolicy));
1874  }
1875 
1876  public function isPasswordExpired(): bool
1877  {
1878  if ($this->id == ANONYMOUS_USER_ID) {
1879  return false;
1880  }
1881 
1882  $security = ilSecuritySettings::_getInstance();
1883  if ($this->getLastPasswordChangeTS() > 0) {
1884  $max_pass_age = $security->getPasswordMaxAge();
1885  if ($max_pass_age > 0) {
1886  $max_pass_age_ts = ($max_pass_age * 86400);
1887  $pass_change_ts = $this->getLastPasswordChangeTS();
1888  $current_ts = time();
1889 
1890  if (($current_ts - $pass_change_ts) > $max_pass_age_ts) {
1892  return true;
1893  }
1894  }
1895  }
1896  }
1897 
1898  return false;
1899  }
1900 
1901  public function getPasswordAge(): int
1902  {
1903  $current_ts = time();
1904  $pass_change_ts = $this->getLastPasswordChangeTS();
1905  $password_age = (int) (($current_ts - $pass_change_ts) / 86400);
1906  return $password_age;
1907  }
1908 
1909  public function setLastPasswordChangeToNow(): bool
1910  {
1911  global $DIC;
1912 
1913  $ilDB = $DIC['ilDB'];
1914 
1915  $this->setLastPasswordChangeTS(time());
1916 
1917  $query = 'UPDATE usr_data SET last_password_change = %s ' .
1918  'WHERE usr_id = %s';
1919  $affected = $ilDB->manipulateF(
1920  $query,
1921  ['integer','integer'],
1922  [$this->getLastPasswordChangeTS(),$this->id]
1923  );
1924  if ($affected) {
1925  return true;
1926  } else {
1927  return false;
1928  }
1929  }
1930 
1931  public function resetLastPasswordChange(): bool
1932  {
1933  $ilDB = $this->db;
1934 
1935  $query = 'UPDATE usr_data SET last_password_change = 0 ' .
1936  'WHERE usr_id = %s';
1937  $affected = $ilDB->manipulateF(
1938  $query,
1939  ['integer'],
1940  [$this->getId()]
1941  );
1942  if ($affected) {
1943  return true;
1944  } else {
1945  return false;
1946  }
1947  }
1948 
1949  public function setLatitude(?string $a_latitude): void
1950  {
1951  $this->latitude = $a_latitude;
1952  }
1953 
1954  public function getLatitude(): ?string
1955  {
1956  return $this->latitude;
1957  }
1958 
1959  public function setLongitude(?string $a_longitude): void
1960  {
1961  $this->longitude = $a_longitude;
1962  }
1963 
1964  public function getLongitude(): ?string
1965  {
1966  return $this->longitude;
1967  }
1968 
1969  public function setLocationZoom(?int $a_locationzoom): void
1970  {
1971  $this->loc_zoom = $a_locationzoom;
1972  }
1973 
1974  public function getLocationZoom(): ?int
1975  {
1976  return $this->loc_zoom;
1977  }
1978 
1979  public function getAvatarRid(): ?string
1980  {
1981  return $this->avatar_rid;
1982  }
1983 
1984  public function setAvatarRid(?string $avatar_rid): void
1985  {
1986  $this->avatar_rid = $avatar_rid;
1987  }
1988 
1989 
1990  public static function hasActiveSession(
1991  int $a_user_id,
1992  string $a_session_id
1993  ): bool {
1994  global $DIC;
1995 
1996  $ilDB = $DIC['ilDB'];
1997 
1998  $set = $ilDB->queryf(
1999  '
2000  SELECT COUNT(*) session_count
2001  FROM usr_session WHERE user_id = %s AND expires > %s AND session_id != %s ',
2002  ['integer', 'integer', 'text'],
2003  [$a_user_id, time(), $a_session_id]
2004  );
2005  $row = $ilDB->fetchAssoc($set);
2006  return (bool) $row['session_count'];
2007  }
2008 
2012  public function checkUserId(): bool
2013  {
2014  $login = self::getLoginFromAuth();
2015  $id = self::_lookupId($login);
2016  if ($id > 0) {
2017  return $id;
2018  }
2019  return false;
2020  }
2021 
2025  private static function getLoginFromAuth(): string
2026  {
2027  $uid = $GLOBALS['DIC']['ilAuthSession']->getUserId();
2028  $login = self::_lookupLogin($uid);
2029 
2030  // BEGIN WebDAV: Strip Microsoft Domain Names from logins
2032  $login = self::toUsernameWithoutDomain($login);
2033  }
2034  return $login;
2035  }
2036 
2041  public static function toUsernameWithoutDomain(string $a_login): string
2042  {
2043  // Remove all characters including the last slash or the last backslash
2044  // in the username
2045  $pos = strrpos($a_login, '/');
2046  $pos2 = strrpos($a_login, '\\');
2047  if ($pos === false || $pos < $pos2) {
2048  $pos = $pos2;
2049  }
2050  if (is_int($pos)) {
2051  $a_login = substr($a_login, $pos + 1);
2052  }
2053  return $a_login;
2054  }
2055 
2056  /*
2057  * check to see if current user has been made active
2058  */
2059  public function isCurrentUserActive(): bool
2060  {
2061  $ilDB = $this->db;
2062 
2063  $login = self::getLoginFromAuth();
2064  $set = $ilDB->queryF(
2065  'SELECT active FROM usr_data WHERE login= %s',
2066  ['text'],
2067  [$login]
2068  );
2069  //query has got a result
2070  if ($rec = $ilDB->fetchAssoc($set)) {
2071  if ($rec['active']) {
2072  return true;
2073  }
2074  }
2075 
2076  return false;
2077  }
2078 
2079  public static function getUserIdByLogin(string $a_login): int
2080  {
2081  return (int) self::_lookupId($a_login);
2082  }
2083 
2087  public static function getUserIdsByEmail(string $a_email): array
2088  {
2089  global $DIC;
2090 
2091  $ilDB = $DIC->database();
2092 
2093  $res = $ilDB->queryF(
2094  'SELECT usr_id FROM usr_data ' .
2095  'WHERE email = %s and active = 1',
2096  ['text'],
2097  [$a_email]
2098  );
2099  $ids = [];
2100  while ($row = $ilDB->fetchObject($res)) {
2101  $ids[] = (int) $row->usr_id;
2102  }
2103 
2104  return $ids;
2105  }
2106 
2107 
2111  public static function getUserLoginsByEmail(string $a_email): array
2112  {
2113  global $DIC;
2114 
2115  $ilDB = $DIC->database();
2116 
2117  $res = $ilDB->queryF(
2118  'SELECT login FROM usr_data ' .
2119  'WHERE email = %s and active = 1',
2120  ['text'],
2121  [$a_email]
2122  );
2123  $ids = [];
2124  while ($row = $ilDB->fetchObject($res)) {
2125  $ids[] = $row->login;
2126  }
2127 
2128  return $ids;
2129  }
2130 
2131  public function getLoginByUserId(int $a_userid): ?string
2132  {
2133  $login = self::_lookupLogin($a_userid);
2134  return $login ?: null;
2135  }
2136 
2140  public static function getAllUserLogins(): array
2141  {
2145  global $DIC;
2146 
2147  $ilDB = $DIC['ilDB'];
2148 
2149  $logins = [];
2150 
2151  $res = $ilDB->query(
2152  'SELECT login FROM usr_data WHERE ' . $ilDB->in('usr_id', [ANONYMOUS_USER_ID], true, 'integer')
2153  );
2154  while ($row = $ilDB->fetchAssoc($res)) {
2155  $logins[] = $row['login'];
2156  }
2157 
2158  return $logins;
2159  }
2160 
2165  public static function _readUsersProfileData(array $a_user_ids): array
2166  {
2167  global $DIC;
2168 
2169  $ilDB = $DIC['ilDB'];
2170  $res = $ilDB->query('SELECT * FROM usr_data WHERE ' .
2171  $ilDB->in('usr_id', $a_user_ids, false, 'integer'));
2172  $user_data = [];
2173  while ($row = $ilDB->fetchAssoc($res)) {
2174  $user_data[$row['usr_id']] = $row;
2175  }
2176  return $user_data;
2177  }
2178 
2184  public static function _getAllUserData(
2185  ?array $a_fields = null,
2186  int $active = -1
2187  ): array {
2188  global $DIC;
2189 
2190  $ilDB = $DIC['ilDB'];
2191 
2192  $result_arr = [];
2193 
2194  if ($a_fields !== null and is_array($a_fields)) {
2195  if (count($a_fields) == 0) {
2196  $select = '*';
2197  } else {
2198  if (($usr_id_field = array_search('usr_id', $a_fields)) !== false) {
2199  unset($a_fields[$usr_id_field]);
2200  }
2201 
2202  $select = implode(',', $a_fields) . ',usr_data.usr_id';
2203  // online time
2204  if (in_array('online_time', $a_fields)) {
2205  $select .= ',ut_online.online_time ';
2206  }
2207  }
2208 
2209  $q = 'SELECT ' . $select . ' FROM usr_data ';
2210 
2211  // Add online_time if desired
2212  // Need left join here to show users that never logged in
2213  if (in_array('online_time', $a_fields)) {
2214  $q .= 'LEFT JOIN ut_online ON usr_data.usr_id = ut_online.usr_id ';
2215  }
2216 
2217  switch ($active) {
2218  case 0:
2219  case 1:
2220  $q .= 'WHERE active = ' . $ilDB->quote($active, 'integer');
2221  break;
2222  case 2:
2223  $q .= 'WHERE time_limit_unlimited= ' . $ilDB->quote(0, 'integer');
2224  break;
2225  case 3:
2226  $qtemp = $q . ', rbac_ua, object_data WHERE rbac_ua.rol_id = object_data.obj_id AND ' .
2227  $ilDB->like('object_data.title', 'text', '%crs%') . ' AND usr_data.usr_id = rbac_ua.usr_id';
2228  $r = $ilDB->query($qtemp);
2229  $course_users = [];
2230  while ($row = $ilDB->fetchAssoc($r)) {
2231  $course_users[] = $row['usr_id'];
2232  }
2233  if (count($course_users)) {
2234  $q .= ' WHERE ' . $ilDB->in('usr_data.usr_id', $course_users, true, 'integer') . ' ';
2235  } else {
2236  return $result_arr;
2237  }
2238  break;
2239  case 4:
2240  $session_data = ilSession::get('user_filter_data');
2241  $date = date('Y-m-d H:i:s', mktime(0, 0, 0, $session_data['m'], $session_data['d'], $session_data['y']));
2242  $q .= ' AND last_login < ' . $ilDB->quote($date, 'timestamp');
2243  break;
2244  case 5:
2245  $ref_id = ilSession::get('user_filter_data');
2246  if ($ref_id) {
2247  $q .= ' LEFT JOIN obj_members ON usr_data.usr_id = obj_members.usr_id ' .
2248  'WHERE obj_members.obj_id = (SELECT obj_id FROM object_reference ' .
2249  'WHERE ref_id = ' . $ilDB->quote($ref_id, 'integer') . ') ';
2250  }
2251  break;
2252  case 6:
2253  global $DIC;
2254 
2255  $rbacreview = $DIC['rbacreview'];
2256  $ref_id = ilSession::get('user_filter_data');
2257  if ($ref_id) {
2258  $local_roles = $rbacreview->getRolesOfRoleFolder($ref_id, false);
2259  if (is_array($local_roles) && count($local_roles)) {
2260  $q .= ' LEFT JOIN rbac_ua ON usr_data.usr_id = rbac_ua.usr_id WHERE ' .
2261  $ilDB->in('rbac_ua.rol_id', $local_roles, false, 'integer') . ' ';
2262  }
2263  }
2264  break;
2265  case 7:
2266  $rol_id = ilSession::get('user_filter_data');
2267  if ($rol_id) {
2268  $q .= ' LEFT JOIN rbac_ua ON usr_data.usr_id = rbac_ua.usr_id WHERE rbac_ua.rol_id = ' .
2269  $ilDB->quote($rol_id, 'integer');
2270  }
2271  break;
2272  }
2273  $r = $ilDB->query($q);
2274 
2275  while ($row = $ilDB->fetchAssoc($r)) {
2276  $result_arr[] = $row;
2277  }
2278  }
2279 
2280  return $result_arr;
2281  }
2282 
2283  public static function _getNumberOfUsersForStyle(
2284  string $a_skin,
2285  string $a_style
2286  ): int {
2287  global $DIC;
2288 
2289  $ilDB = $DIC['ilDB'];
2290 
2291  $q = 'SELECT count(*) as cnt FROM usr_pref up1, usr_pref up2 ' .
2292  ' WHERE up1.keyword= ' . $ilDB->quote('style', 'text') .
2293  ' AND up1.value= ' . $ilDB->quote($a_style, 'text') .
2294  ' AND up2.keyword= ' . $ilDB->quote('skin', 'text') .
2295  ' AND up2.value= ' . $ilDB->quote($a_skin, 'text') .
2296  ' AND up1.usr_id = up2.usr_id ';
2297 
2298  $cnt_set = $ilDB->query($q);
2299 
2300  $cnt_rec = $ilDB->fetchAssoc($cnt_set);
2301 
2302  return (int) $cnt_rec['cnt'];
2303  }
2304 
2308  public static function _getAllUserAssignedStyles(): array
2309  {
2310  global $DIC;
2311 
2312  $ilDB = $DIC['ilDB'];
2313 
2314  $q = 'SELECT DISTINCT up1.value style, up2.value skin FROM usr_pref up1, usr_pref up2 ' .
2315  ' WHERE up1.keyword = ' . $ilDB->quote('style', 'text') .
2316  ' AND up2.keyword = ' . $ilDB->quote('skin', 'text') .
2317  ' AND up1.usr_id = up2.usr_id';
2318 
2319  $sty_set = $ilDB->query($q);
2320 
2321  $styles = [];
2322  while ($sty_rec = $ilDB->fetchAssoc($sty_set)) {
2323  $styles[] = $sty_rec['skin'] . ':' . $sty_rec['style'];
2324  }
2325 
2326  return $styles;
2327  }
2328 
2329  public static function _moveUsersToStyle(
2330  string $a_from_skin,
2331  string $a_from_style,
2332  string $a_to_skin,
2333  string $a_to_style
2334  ): void {
2335  global $DIC;
2336 
2337  $ilDB = $DIC['ilDB'];
2338 
2339  $q = 'SELECT up1.usr_id usr_id FROM usr_pref up1, usr_pref up2 ' .
2340  ' WHERE up1.keyword= ' . $ilDB->quote('style', 'text') .
2341  ' AND up1.value= ' . $ilDB->quote($a_from_style, 'text') .
2342  ' AND up2.keyword= ' . $ilDB->quote('skin', 'text') .
2343  ' AND up2.value= ' . $ilDB->quote($a_from_skin, 'text') .
2344  ' AND up1.usr_id = up2.usr_id ';
2345 
2346  $usr_set = $ilDB->query($q);
2347 
2348  while ($usr_rec = $ilDB->fetchAssoc($usr_set)) {
2349  self::_writePref($usr_rec['usr_id'], 'skin', $a_to_skin);
2350  self::_writePref($usr_rec['usr_id'], 'style', $a_to_style);
2351  }
2352  }
2353 
2354 
2360 
2367  public function addObjectToClipboard(
2368  int $a_item_id,
2369  string $a_type,
2370  string $a_title,
2371  int $a_parent = 0,
2372  string $a_time = '',
2373  int $a_order_nr = 0
2374  ): void {
2375  global $DIC;
2376 
2377  $ilDB = $DIC['ilDB'];
2378 
2379  if ($a_time === '') {
2380  $a_time = date('Y-m-d H:i:s');
2381  }
2382 
2383  $item_set = $ilDB->queryF(
2384  'SELECT * FROM personal_clipboard WHERE ' .
2385  'parent = %s AND item_id = %s AND type = %s AND user_id = %s',
2386  ['integer', 'integer', 'text', 'integer'],
2387  [0, $a_item_id, $a_type, $this->getId()]
2388  );
2389 
2390  // only insert if item is not already in clipboard
2391  if (!$item_set->fetchRow()) {
2392  $ilDB->manipulateF(
2393  'INSERT INTO personal_clipboard ' .
2394  '(item_id, type, user_id, title, parent, insert_time, order_nr) VALUES ' .
2395  ' (%s,%s,%s,%s,%s,%s,%s)',
2396  ['integer', 'text', 'integer', 'text', 'integer', 'timestamp', 'integer'],
2397  [$a_item_id, $a_type, $this->getId(), $a_title, $a_parent, $a_time, $a_order_nr]
2398  );
2399  } else {
2400  $ilDB->manipulateF(
2401  'UPDATE personal_clipboard SET insert_time = %s ' .
2402  'WHERE user_id = %s AND item_id = %s AND type = %s AND parent = 0',
2403  ['timestamp', 'integer', 'integer', 'text'],
2404  [$a_time, $this->getId(), $a_item_id, $a_type]
2405  );
2406  }
2407  }
2408 
2413  public function addToPCClipboard(
2414  string $a_content,
2415  string $a_time,
2416  int $a_nr
2417  ): void {
2418  $ilDB = $this->db;
2419  if ($a_time == 0) {
2420  $a_time = date('Y-m-d H:i:s');
2421  }
2422  ilSession::set('user_pc_clip', true);
2423  $ilDB->insert('personal_pc_clipboard', [
2424  'user_id' => ['integer', $this->getId()],
2425  'content' => ['clob', $a_content],
2426  'insert_time' => ['timestamp', $a_time],
2427  'order_nr' => ['integer', $a_nr]
2428  ]);
2429  }
2430 
2435  public function getPCClipboardContent(): array // Missing array type.
2436  {
2437  $ilDB = $this->db;
2438 
2439  if (!ilSession::get('user_pc_clip')) {
2440  return [];
2441  }
2442 
2443  $set = $ilDB->queryF('SELECT MAX(insert_time) mtime FROM personal_pc_clipboard ' .
2444  ' WHERE user_id = %s', ['integer'], [$this->getId()]);
2445  $row = $ilDB->fetchAssoc($set);
2446 
2447  $set = $ilDB->queryF(
2448  'SELECT * FROM personal_pc_clipboard ' .
2449  ' WHERE user_id = %s AND insert_time = %s ORDER BY order_nr ASC',
2450  ['integer', 'timestamp'],
2451  [$this->getId(), $row['mtime']]
2452  );
2453  $content = [];
2454  while ($row = $ilDB->fetchAssoc($set)) {
2455  $content[] = $row['content'];
2456  }
2457 
2458  return $content;
2459  }
2460 
2464  public function clipboardHasObjectsOfType(string $a_type): bool
2465  {
2466  global $DIC;
2467 
2468  $ilDB = $DIC['ilDB'];
2469 
2470  $set = $ilDB->queryF(
2471  'SELECT * FROM personal_clipboard WHERE ' .
2472  'parent = %s AND type = %s AND user_id = %s',
2473  ['integer', 'text', 'integer'],
2474  [0, $a_type, $this->getId()]
2475  );
2476  if ($ilDB->fetchAssoc($set)) {
2477  return true;
2478  }
2479 
2480  return false;
2481  }
2482 
2483  public function clipboardDeleteObjectsOfType(string $a_type): void
2484  {
2485  $ilDB = $this->db;
2486 
2487  $ilDB->manipulateF(
2488  'DELETE FROM personal_clipboard WHERE ' .
2489  'type = %s AND user_id = %s',
2490  ['text', 'integer'],
2491  [$a_type, $this->getId()]
2492  );
2493  }
2494 
2495  public function clipboardDeleteAll(): void
2496  {
2497  global $DIC;
2498 
2499  $ilDB = $DIC['ilDB'];
2500 
2501  $ilDB->manipulateF('DELETE FROM personal_clipboard WHERE ' .
2502  'user_id = %s', ['integer'], [$this->getId()]);
2503  }
2504 
2508  public function getClipboardObjects(
2509  string $a_type = '',
2510  bool $a_top_nodes_only = false
2511  ): array {
2512  global $DIC;
2513 
2514  $ilDB = $DIC['ilDB'];
2515 
2516  $par = '';
2517  if ($a_top_nodes_only) {
2518  $par = ' AND parent = ' . $ilDB->quote(0, 'integer') . ' ';
2519  }
2520 
2521  $type_str = ($a_type != '')
2522  ? ' AND type = ' . $ilDB->quote($a_type, 'text') . ' '
2523  : '';
2524  $q = 'SELECT * FROM personal_clipboard WHERE ' .
2525  'user_id = ' . $ilDB->quote($this->getId(), 'integer') . ' ' .
2526  $type_str . $par .
2527  ' ORDER BY order_nr';
2528  $objs = $ilDB->query($q);
2529  $objects = [];
2530  while ($obj = $ilDB->fetchAssoc($objs)) {
2531  if ($obj['type'] == 'mob') {
2532  $obj['title'] = ilObject::_lookupTitle($obj['item_id']);
2533  if (ilObject::_lookupType((int) $obj['item_id']) !== 'mob') {
2534  continue;
2535  }
2536  }
2537  if ($obj['type'] == 'incl') {
2538  $obj['title'] = ilMediaPoolPage::lookupTitle($obj['item_id']);
2539  if (!ilPageObject::_exists('mep', (int) $obj['item_id'], '-')) {
2540  continue;
2541  }
2542  }
2543  $objects[] = ['id' => $obj['item_id'],
2544  'type' => $obj['type'], 'title' => $obj['title'],
2545  'insert_time' => $obj['insert_time']];
2546  }
2547  return $objects;
2548  }
2549 
2553  public function getClipboardChilds(
2554  int $a_parent,
2555  string $a_insert_time
2556  ): array {
2557  global $DIC;
2558 
2559  $ilDB = $DIC['ilDB'];
2560  $ilUser = $DIC['ilUser'];
2561 
2562  $objs = $ilDB->queryF(
2563  'SELECT * FROM personal_clipboard WHERE ' .
2564  'user_id = %s AND parent = %s AND insert_time = %s ' .
2565  ' ORDER BY order_nr',
2566  ['integer', 'integer', 'timestamp'],
2567  [$ilUser->getId(), $a_parent, $a_insert_time]
2568  );
2569  $objects = [];
2570  while ($obj = $ilDB->fetchAssoc($objs)) {
2571  if ($obj['type'] == 'mob') {
2572  $obj['title'] = ilObject::_lookupTitle($obj['item_id']);
2573  }
2574  $objects[] = ['id' => $obj['item_id'],
2575  'type' => $obj['type'], 'title' => $obj['title'], 'insert_time' => $obj['insert_time']];
2576  }
2577  return $objects;
2578  }
2579 
2584  public static function _getUsersForClipboadObject(
2585  string $a_type,
2586  int $a_id
2587  ): array {
2588  global $DIC;
2589 
2590  $ilDB = $DIC['ilDB'];
2591 
2592  $q = 'SELECT DISTINCT user_id FROM personal_clipboard WHERE ' .
2593  'item_id = ' . $ilDB->quote($a_id, 'integer') . ' AND ' .
2594  'type = ' . $ilDB->quote($a_type, 'text');
2595  $user_set = $ilDB->query($q);
2596  $users = [];
2597  while ($user_rec = $ilDB->fetchAssoc($user_set)) {
2598  $users[] = (int) $user_rec['user_id'];
2599  }
2600 
2601  return $users;
2602  }
2603 
2604  public function removeObjectFromClipboard(
2605  int $a_item_id,
2606  string $a_type
2607  ): void {
2608  $ilDB = $this->db;
2609 
2610  $q = 'DELETE FROM personal_clipboard WHERE ' .
2611  'item_id = ' . $ilDB->quote($a_item_id, 'integer') .
2612  ' AND type = ' . $ilDB->quote($a_type, 'text') . ' ' .
2613  ' AND user_id = ' . $ilDB->quote($this->getId(), 'integer');
2614  $ilDB->manipulate($q);
2615  }
2616 
2617  public static function _getImportedUserId(
2618  string $i2_id
2619  ): int {
2620  global $DIC;
2621 
2622  $ilDB = $DIC['ilDB'];
2623 
2624  $query = 'SELECT obj_id FROM object_data WHERE import_id = ' .
2625  $ilDB->quote($i2_id, 'text');
2626 
2627  $res = $ilDB->query($query);
2628  $id = 0;
2629  while ($row = $ilDB->fetchObject($res)) {
2630  $id = (int) $row->obj_id;
2631  }
2632  return $id;
2633  }
2634 
2638  public static function lookupOrgUnitsRepresentation(
2639  int $a_usr_id
2640  ): string {
2641  return ilOrgUnitPathStorage::getTextRepresentationOfUsersOrgUnits($a_usr_id);
2642  }
2643 
2644  public function getOrgUnitsRepresentation(): string
2645  {
2646  return self::lookupOrgUnitsRepresentation($this->getId());
2647  }
2648 
2649  public function setAuthMode(?string $a_str): void
2650  {
2651  $this->auth_mode = $a_str;
2652  }
2653 
2654  public function getAuthMode(bool $a_auth_key = false): ?string
2655  {
2656  if (!$a_auth_key) {
2657  return $this->auth_mode;
2658  }
2659  return ilAuthUtils::_getAuthMode($this->auth_mode);
2660  }
2661 
2662  public function setExternalAccount(string $a_str): void
2663  {
2664  $this->ext_account = $a_str;
2665  }
2666 
2667  public function getExternalAccount(): string
2668  {
2669  return $this->ext_account;
2670  }
2671 
2678  public static function _getExternalAccountsByAuthMode(
2679  string $a_auth_mode,
2680  bool $a_read_auth_default = false
2681  ): array {
2682  global $DIC;
2683 
2684  $ilDB = $DIC['ilDB'];
2685  $ilSetting = $DIC['ilSetting'];
2686 
2687  $q = 'SELECT login,usr_id,ext_account,auth_mode FROM usr_data ' .
2688  'WHERE auth_mode = %s';
2689  $types[] = 'text';
2690  $values[] = $a_auth_mode;
2691  if ($a_read_auth_default and ilAuthUtils::_getAuthModeName($ilSetting->get('auth_mode', ilAuthUtils::AUTH_LOCAL)) == $a_auth_mode) {
2692  $q .= ' OR auth_mode = %s ';
2693  $types[] = 'text';
2694  $values[] = 'default';
2695  }
2696 
2697  $res = $ilDB->queryF($q, $types, $values);
2698  $accounts = [];
2699  while ($row = $ilDB->fetchObject($res)) {
2700  if ($row->auth_mode == 'default') {
2701  $accounts[$row->usr_id] = $row->login;
2702  } else {
2703  $accounts[$row->usr_id] = $row->ext_account;
2704  }
2705  }
2706  return $accounts;
2707  }
2708 
2709  public static function _toggleActiveStatusOfUsers(
2710  array $a_usr_ids,
2711  bool $a_status
2712  ): void {
2713  global $DIC;
2714 
2715  $ilDB = $DIC['ilDB'];
2716 
2717  if ($a_status) {
2718  $q = 'UPDATE usr_data SET active = 1, inactivation_date = NULL WHERE ' .
2719  $ilDB->in('usr_id', $a_usr_ids, false, 'integer');
2720  $ilDB->manipulate($q);
2721  } else {
2722  $usrId_IN_usrIds = $ilDB->in('usr_id', $a_usr_ids, false, 'integer');
2723 
2724  $q = 'UPDATE usr_data SET active = 0 WHERE $usrId_IN_usrIds';
2725  $ilDB->manipulate($q);
2726 
2727  $queryString = '
2728  UPDATE usr_data
2729  SET inactivation_date = %s
2730  WHERE inactivation_date IS NULL
2731  AND $usrId_IN_usrIds
2732  ';
2733  $ilDB->manipulateF($queryString, ['timestamp'], [ilUtil::now()]);
2734  }
2735  }
2736 
2737  public static function _lookupAuthMode(int $a_usr_id): string
2738  {
2739  return (string) self::_lookup($a_usr_id, 'auth_mode');
2740  }
2741 
2746  public static function _checkExternalAuthAccount(
2747  string $a_auth,
2748  string $a_account,
2749  bool $tryFallback = true
2750  ): ?string {
2751  $db = $GLOBALS['DIC']->database();
2752  $settings = $GLOBALS['DIC']->settings();
2753 
2754  // Check directly with auth_mode
2755  $r = $db->queryF(
2756  'SELECT * FROM usr_data WHERE ' .
2757  ' ext_account = %s AND auth_mode = %s',
2758  ['text', 'text'],
2759  [$a_account, $a_auth]
2760  );
2761  if ($usr = $db->fetchAssoc($r)) {
2762  return $usr['login'];
2763  }
2764 
2765  if (!$tryFallback) {
2766  return null;
2767  }
2768 
2769  // For compatibility, check for login (no ext_account entry given)
2770  $res = $db->queryF(
2771  'SELECT login FROM usr_data ' .
2772  'WHERE login = %s AND auth_mode = %s AND (ext_account IS NULL OR ext_account = "") ',
2773  ['text', 'text'],
2774  [$a_account, $a_auth]
2775  );
2776  if ($usr = $db->fetchAssoc($res)) {
2777  return $usr['login'];
2778  }
2779 
2780  // If auth_default == $a_auth => check for login
2781  if (ilAuthUtils::_getAuthModeName($settings->get('auth_mode')) == $a_auth) {
2782  $res = $db->queryF(
2783  'SELECT login FROM usr_data WHERE ' .
2784  ' ext_account = %s AND auth_mode = %s',
2785  ['text', 'text'],
2786  [$a_account, 'default']
2787  );
2788  if ($usr = $db->fetchAssoc($res)) {
2789  return $usr['login'];
2790  }
2791  // Search for login (no ext_account given)
2792  $res = $db->queryF(
2793  'SELECT login FROM usr_data ' .
2794  'WHERE login = %s AND (ext_account IS NULL OR ext_account = "") AND auth_mode = %s',
2795  ['text', 'text'],
2796  [$a_account, 'default']
2797  );
2798  if ($usr = $db->fetchAssoc($res)) {
2799  return $usr['login'];
2800  }
2801  }
2802  return null;
2803  }
2804 
2808  public static function _getNumberOfUsersPerAuthMode(): array // Missing array type.
2809  {
2810  global $DIC;
2811 
2812  $ilDB = $DIC['ilDB'];
2813 
2814  $r = $ilDB->query('SELECT count(*) AS cnt, auth_mode FROM usr_data ' .
2815  'GROUP BY auth_mode');
2816  $cnt_arr = [];
2817  while ($cnt = $ilDB->fetchAssoc($r)) {
2818  $cnt_arr[$cnt['auth_mode']] = (int) $cnt['cnt'];
2819  }
2820 
2821  return $cnt_arr;
2822  }
2823 
2824  public static function _getLocalAccountsForEmail(string $a_email): array // Missing array type.
2825  {
2826  global $DIC;
2827 
2828  $ilDB = $DIC['ilDB'];
2829  $ilSetting = $DIC['ilSetting'];
2830 
2831  // default set to local (1)?
2832 
2833  $q = 'SELECT * FROM usr_data WHERE ' .
2834  ' email = %s AND (auth_mode = %s ';
2835  $types = ['text', 'text'];
2836  $values = [$a_email, 'local'];
2837 
2838  if ($ilSetting->get('auth_mode') == 1) {
2839  $q .= ' OR auth_mode = %s';
2840  $types[] = 'text';
2841  $values[] = 'default';
2842  }
2843 
2844  $q .= ')';
2845 
2846  $users = [];
2847  $usr_set = $ilDB->queryF($q, $types, $values);
2848  while ($usr_rec = $ilDB->fetchAssoc($usr_set)) {
2849  $users[$usr_rec['usr_id']] = $usr_rec['login'];
2850  }
2851 
2852  return $users;
2853  }
2854 
2855 
2860  public static function _uploadPersonalPicture(
2861  string $tmp_file,
2862  int $obj_id
2863  ): bool {
2864  global $DIC;
2865  $user = new ilObjUser($obj_id);
2866  $stakeholder = new ilUserProfilePictureStakeholder();
2867  $stakeholder->setOwner($user->getId());
2868  $stream = Streams::ofResource(fopen($tmp_file, 'rb'));
2869 
2871  $rid = $DIC->resourceStorage()->manage()->find($user->getAvatarRid());
2872  // append profile picture
2873  $DIC->resourceStorage()->manage()->replaceWithStream(
2874  $rid,
2875  $stream,
2876  $stakeholder
2877  );
2878  } else {
2879  // new profile picture
2880  $rid = $DIC->resourceStorage()->manage()->stream(
2881  $stream,
2882  $stakeholder
2883  );
2884  }
2885 
2886  $user->setAvatarRid($rid->serialize());
2887  $user->update();
2888  return true;
2889  }
2890 
2891 
2896  public function getPersonalPicturePath(
2897  string $a_size = 'small',
2898  bool $a_force_pic = false
2899  ): string {
2900  if (isset(self::$personal_image_cache[$this->getId()][$a_size][(int) $a_force_pic])) {
2901  return self::$personal_image_cache[$this->getId()][$a_size][(int) $a_force_pic];
2902  }
2903 
2904  self::$personal_image_cache[$this->getId()][$a_size][(int) $a_force_pic] = self::_getPersonalPicturePath($this->getId(), $a_size, $a_force_pic);
2905 
2906  return self::$personal_image_cache[$this->getId()][$a_size][(int) $a_force_pic];
2907  }
2908 
2909  public function hasProfilePicture(): bool
2910  {
2911  return (new ilUserAvatarResolver($this->getId()))->hasProfilePicture();
2912  }
2913 
2914  public function getAvatar(): Avatar
2915  {
2916  return self::_getAvatar($this->getId());
2917  }
2918 
2919  public static function _getAvatar(int $a_usr_id): Avatar
2920  {
2921  $define = new ilUserAvatarResolver($a_usr_id ?: ANONYMOUS_USER_ID);
2922  $define->setSize('xsmall');
2923  return $define->getAvatar();
2924  }
2925 
2930  public static function _getPersonalPicturePath(
2931  int $a_usr_id,
2932  string $a_size = 'small',
2933  bool $a_force_pic = false,
2934  bool $a_prevent_no_photo_image = false,
2935  bool $html_export = false
2936  ): string {
2937  $define = new ilUserAvatarResolver($a_usr_id);
2938  $define->setForcePicture($a_force_pic);
2939  $define->setSize($a_size);
2940  return $define->getLegacyPictureURL();
2941  }
2942 
2943  public static function copyProfilePicturesToDirectory(
2944  int $a_user_id,
2945  string $a_dir
2946  ): void {
2947  $a_dir = trim(str_replace('..', '', $a_dir));
2948  if ($a_dir == '' || !is_dir($a_dir)) {
2949  return;
2950  }
2951  // if profile picture is on IRSS
2952  global $DIC;
2953  $irss = $DIC->resourceStorage();
2954  $user = new ilObjUser($a_user_id);
2956  $rid = $irss->manage()->find($user->getAvatarRid());
2957  // Main Picture only is needed
2958  $stream = $irss->consume()->stream($rid)->getStream();
2959  $target = $a_dir . '/usr_' . $a_user_id . '.jpg';
2960  fwrite(fopen($target, 'wb'), (string) $stream);
2961 
2962  return;
2963  }
2964 
2965  // Legacy Picture Handling
2966  $webspace_dir = ilFileUtils::getWebspaceDir();
2967  $image_dir = $webspace_dir . '/usr_images';
2968  $images = [
2969  'upload_' . $a_user_id . 'pic',
2970  'usr_' . $a_user_id . '.' . 'jpg',
2971  'usr_' . $a_user_id . '_small.jpg',
2972  'usr_' . $a_user_id . '_xsmall.jpg',
2973  'usr_' . $a_user_id . '_xxsmall.jpg',
2974  'upload_' . $a_user_id];
2975  foreach ($images as $image) {
2976  if (is_file($image_dir . '/' . $image)) {
2977  copy($image_dir . '/' . $image, $a_dir . '/' . $image);
2978  }
2979  }
2980  }
2981 
2982 
2983  public function removeUserPicture(
2984  bool $a_do_update = true
2985  ): void {
2986  if ($this->getAvatarRid() !== null && $this->getAvatarRid() !== self::NO_AVATAR_RID) {
2987  $rid = $this->irss->manage()->find($this->getAvatarRid());
2988  $this->irss->manage()->remove($rid, new ilUserProfilePictureStakeholder());
2989  }
2990 
2991  if ($a_do_update) {
2992  // remove user pref file name
2994  $this->setPref('profile_image', '');
2995  $this->update();
2996  }
2997  }
2998 
2999 
3000  public function setUserDefinedData(array $a_data): void // Missing array type.
3001  {
3002  foreach ($a_data as $field => $data) {
3003  $this->user_defined_data['f_' . $field] = $data;
3004  }
3005  }
3006 
3007  public function getUserDefinedData(): array // Missing array type.
3008  {
3009  return $this->user_defined_data ?: [];
3010  }
3011 
3012  public function updateUserDefinedFields(): void
3013  {
3014  $udata = new ilUserDefinedData($this->getId());
3015  foreach ($this->user_defined_data as $field => $value) {
3016  if ($field !== 'usr_id' && $value !== null) {
3017  $udata->set($field, $value);
3018  }
3019  }
3020  $udata->update();
3021  }
3022 
3023  public function readUserDefinedFields(): void
3024  {
3025  $udata = new ilUserDefinedData($this->getId());
3026  $this->user_defined_data = $udata->getAll();
3027  }
3028 
3029  public function deleteUserDefinedFieldEntries(): void
3030  {
3032  }
3033 
3038  public function getProfileAsString(Language $language): string
3039  {
3040  global $DIC;
3041 
3042  $rbacreview = $DIC['rbacreview'];
3043 
3044  $language->loadLanguageModule('registration');
3045  $language->loadLanguageModule('crs');
3046 
3047  $body = ($language->txt('login') . ': ' . $this->getLogin() . "\n");
3048 
3049  if (strlen($this->getUTitle())) {
3050  $body .= ($language->txt('title') . ': ' . $this->getUTitle() . "\n");
3051  }
3052  if (1 === strlen($this->getGender())) {
3053  $body .= ($language->txt('gender') . ': ' . $language->txt('gender_' . strtolower($this->getGender())) . "\n");
3054  }
3055  if (strlen($this->getFirstname())) {
3056  $body .= ($language->txt('firstname') . ': ' . $this->getFirstname() . "\n");
3057  }
3058  if (strlen($this->getLastname())) {
3059  $body .= ($language->txt('lastname') . ': ' . $this->getLastname() . "\n");
3060  }
3061  if (strlen($this->getInstitution())) {
3062  $body .= ($language->txt('institution') . ': ' . $this->getInstitution() . "\n");
3063  }
3064  if (strlen($this->getDepartment())) {
3065  $body .= ($language->txt('department') . ': ' . $this->getDepartment() . "\n");
3066  }
3067  if (strlen($this->getStreet())) {
3068  $body .= ($language->txt('street') . ': ' . $this->getStreet() . "\n");
3069  }
3070  if (strlen($this->getCity())) {
3071  $body .= ($language->txt('city') . ': ' . $this->getCity() . "\n");
3072  }
3073  if (strlen($this->getZipcode())) {
3074  $body .= ($language->txt('zipcode') . ': ' . $this->getZipcode() . "\n");
3075  }
3076  if (strlen($this->getCountry())) {
3077  $body .= ($language->txt('country') . ': ' . $this->getCountry() . "\n");
3078  }
3079  if (strlen($this->getSelectedCountry())) {
3080  $body .= ($language->txt('sel_country') . ': ' . $this->getSelectedCountry() . "\n");
3081  }
3082  if (strlen($this->getPhoneOffice())) {
3083  $body .= ($language->txt('phone_office') . ': ' . $this->getPhoneOffice() . "\n");
3084  }
3085  if (strlen($this->getPhoneHome())) {
3086  $body .= ($language->txt('phone_home') . ': ' . $this->getPhoneHome() . "\n");
3087  }
3088  if (strlen($this->getPhoneMobile())) {
3089  $body .= ($language->txt('phone_mobile') . ': ' . $this->getPhoneMobile() . "\n");
3090  }
3091  if (strlen($this->getFax())) {
3092  $body .= ($language->txt('fax') . ': ' . $this->getFax() . "\n");
3093  }
3094  if ($this->getEmail() !== '') {
3095  $body .= ($language->txt('email') . ': ' . $this->getEmail() . "\n");
3096  }
3097  if ($this->getSecondEmail() !== null
3098  && $this->getSecondEmail() !== '') {
3099  $body .= ($language->txt('second_email') . ': ' . $this->getSecondEmail() . "\n");
3100  }
3101  if (strlen($this->getHobby())) {
3102  $body .= ($language->txt('hobby') . ': ' . $this->getHobby() . "\n");
3103  }
3104  if (strlen($this->getComment())) {
3105  $body .= ($language->txt('referral_comment') . ': ' . $this->getComment() . "\n");
3106  }
3107  if (strlen($this->getMatriculation())) {
3108  $body .= ($language->txt('matriculation') . ': ' . $this->getMatriculation() . "\n");
3109  }
3110  if (strlen($this->getCreateDate())) {
3115 
3116  $body .= ($language->txt('create_date') . ': ' . $date . "\n");
3117  }
3118 
3119  $gr = [];
3120  foreach ($rbacreview->getGlobalRoles() as $role) {
3121  if ($rbacreview->isAssigned($this->getId(), $role)) {
3122  $gr[] = ilObjRole::_lookupTitle($role);
3123  }
3124  }
3125  if (count($gr)) {
3126  $body .= ($language->txt('reg_role_info') . ': ' . implode(',', $gr) . "\n");
3127  }
3128 
3129  // Time limit
3130  if ($this->getTimeLimitUnlimited()) {
3131  $body .= ($language->txt('time_limit') . ': ' . $language->txt('crs_unlimited') . "\n");
3132  } else {
3136  new ilDateTime($this->getTimeLimitFrom(), IL_CAL_UNIX),
3138  );
3140 
3141  $start = new ilDateTime($this->getTimeLimitFrom(), IL_CAL_UNIX);
3142  $end = new ilDateTime($this->getTimeLimitUntil(), IL_CAL_UNIX);
3143 
3144  $body .= $language->txt('time_limit') . ': ' .
3145  $language->txt('from') . ' ' .
3146  $start->get(IL_CAL_DATETIME) . ' ';
3147  $body .= $language->txt('to') . ' ' . $end->get(IL_CAL_DATETIME) . "\n";
3148  }
3149 
3153  $user_defined_fields = ilUserDefinedFields::_getInstance();
3154  $user_defined_data = $this->getUserDefinedData();
3155 
3156  foreach ($user_defined_fields->getDefinitions() as $field_id => $definition) {
3157  $data = $user_defined_data['f_' . $field_id] ?? '';
3158  if ($data !== '') {
3159  if ($definition['field_type'] == UDF_TYPE_WYSIWYG) {
3160  $data = strip_tags(
3161  preg_replace('/<br(\s*)?\/?>/i', "\n", $data)
3162  );
3163  }
3164 
3165  $body .= $definition['field_name'] . ': ' . $data . "\n";
3166  }
3167  }
3168 
3169  return $body;
3170  }
3171 
3175  public static function _lookupFeedHash(
3176  int $a_user_id,
3177  bool $a_create = false
3178  ): ?string {
3179  global $DIC;
3180 
3181  $ilDB = $DIC['ilDB'];
3182 
3183  if ($a_user_id > 0) {
3184  $set = $ilDB->queryF(
3185  'SELECT feed_hash from usr_data WHERE usr_id = %s',
3186  ['integer'],
3187  [$a_user_id]
3188  );
3189  if ($rec = $ilDB->fetchAssoc($set)) {
3190  if (strlen($rec['feed_hash']) == 32) {
3191  return $rec['feed_hash'];
3192  } elseif ($a_create) {
3193  $hash = md5(random_int(1, 9999999) + str_replace(' ', '', microtime()));
3194  $ilDB->manipulateF(
3195  'UPDATE usr_data SET feed_hash = %s' .
3196  ' WHERE usr_id = %s',
3197  ['text', 'integer'],
3198  [$hash, $a_user_id]
3199  );
3200  return $hash;
3201  }
3202  }
3203  }
3204  return null;
3205  }
3206 
3212  public static function _getFeedPass(
3213  int $a_user_id
3214  ): ?string {
3215  if ($a_user_id > 0) {
3216  return self::_lookupPref($a_user_id, 'priv_feed_pass');
3217  }
3218  return null;
3219  }
3220 
3225  public static function _setFeedPass(
3226  int $a_user_id,
3227  string $a_password
3228  ): void {
3229  self::_writePref(
3230  $a_user_id,
3231  'priv_feed_pass',
3232  ($a_password == '') ? '' : md5($a_password)
3233  );
3234  }
3235 
3240  public static function _loginExists(
3241  string $a_login,
3242  int $a_user_id = 0
3243  ): ?int {
3244  global $DIC;
3245 
3246  $ilDB = $DIC['ilDB'];
3247 
3248  $q = 'SELECT DISTINCT login, usr_id FROM usr_data ' .
3249  'WHERE login = %s';
3250  $types[] = 'text';
3251  $values[] = $a_login;
3252 
3253  if ($a_user_id != 0) {
3254  $q .= ' AND usr_id != %s ';
3255  $types[] = 'integer';
3256  $values[] = $a_user_id;
3257  }
3258 
3259  $r = $ilDB->queryF($q, $types, $values);
3260 
3261  if ($row = $ilDB->fetchAssoc($r)) {
3262  return (int) $row['usr_id'];
3263  }
3264  return null;
3265  }
3266 
3270  public static function _externalAccountExists(
3271  string $a_external_account,
3272  string $a_auth_mode
3273  ): bool {
3274  global $DIC;
3275 
3276  $ilDB = $DIC['ilDB'];
3277 
3278  $res = $ilDB->queryF(
3279  'SELECT * FROM usr_data ' .
3280  'WHERE ext_account = %s AND auth_mode = %s',
3281  ['text', 'text'],
3282  [$a_external_account, $a_auth_mode]
3283  );
3284  return (bool) $ilDB->fetchAssoc($res);
3285  }
3286 
3291  public static function _getUsersForRole(
3292  int $role_id,
3293  int $active = -1
3294  ): array {
3295  global $DIC;
3296 
3297  $ilDB = $DIC['ilDB'];
3298  $rbacreview = $DIC['rbacreview'];
3299  $ids = $rbacreview->assignedUsers($role_id);
3300 
3301  if (count($ids) == 0) {
3302  $ids = [-1];
3303  }
3304 
3305  $query = 'SELECT usr_data.*, usr_pref.value AS language
3306  FROM usr_data
3307  LEFT JOIN usr_pref ON usr_pref.usr_id = usr_data.usr_id AND usr_pref.keyword = %s
3308  WHERE ' . $ilDB->in('usr_data.usr_id', $ids, false, 'integer');
3309  $values[] = 'language';
3310  $types[] = 'text';
3311 
3312 
3313  if (is_numeric($active) && $active > -1) {
3314  $query .= ' AND usr_data.active = %s';
3315  $values[] = $active;
3316  $types[] = 'integer';
3317  }
3318 
3319  $query .= ' ORDER BY usr_data.lastname, usr_data.firstname ';
3320 
3321  $r = $ilDB->queryF($query, $types, $values);
3322  $data = [];
3323  while ($row = $ilDB->fetchAssoc($r)) {
3324  $data[] = $row;
3325  }
3326  return $data;
3327  }
3328 
3329 
3334  public static function _getUsersForFolder(
3335  int $ref_id,
3336  int $active
3337  ): array {
3338  global $DIC;
3339 
3340  $ilDB = $DIC['ilDB'];
3341  $query = 'SELECT usr_data.*, usr_pref.value AS language FROM usr_data LEFT JOIN usr_pref ON usr_pref.usr_id = usr_data.usr_id and usr_pref.keyword = %s WHERE 1=1';
3342  $types[] = 'text';
3343  $values[] = 'language';
3344 
3345  if (is_numeric($active) && $active > -1) {
3346  $query .= ' AND usr_data.active = %s';
3347  $values[] = $active;
3348  $types[] = 'integer';
3349  }
3350 
3351  if ($ref_id != USER_FOLDER_ID) {
3352  $query .= ' AND usr_data.time_limit_owner = %s';
3353  $values[] = $ref_id;
3354  $types[] = 'integer';
3355  }
3356 
3357  $query .= ' AND usr_data.usr_id != %s ';
3358  $values[] = ANONYMOUS_USER_ID;
3359  $types[] = 'integer';
3360 
3361  $query .= ' ORDER BY usr_data.lastname, usr_data.firstname ';
3362 
3363  $result = $ilDB->queryF($query, $types, $values);
3364  $data = [];
3365  while ($row = $ilDB->fetchAssoc($result)) {
3366  $data[] = $row;
3367  }
3368 
3369  return $data;
3370  }
3371 
3372 
3378  public static function _getUsersForGroup(
3379  array $a_mem_ids,
3380  int $active = -1
3381  ): array {
3382  return self::_getUsersForIds($a_mem_ids, $active);
3383  }
3384 
3385 
3391  public static function _getUsersForIds(
3392  array $a_mem_ids,
3393  int $active = -1,
3394  int $timelimitowner = -1
3395  ): array {
3396  global $DIC;
3397 
3398  $ilDB = $DIC['ilDB'];
3399 
3400  $query = 'SELECT usr_data.*, usr_pref.value AS language
3401  FROM usr_data
3402  LEFT JOIN usr_pref ON usr_pref.usr_id = usr_data.usr_id AND usr_pref.keyword = %s
3403  WHERE ' . $ilDB->in('usr_data.usr_id', $a_mem_ids, false, 'integer') . '
3404  AND usr_data.usr_id != %s';
3405  $values[] = 'language';
3406  $types[] = 'text';
3407  $values[] = ANONYMOUS_USER_ID;
3408  $types[] = 'integer';
3409 
3410  if (is_numeric($active) && $active > -1) {
3411  $query .= ' AND active = %s';
3412  $values[] = $active;
3413  $types[] = 'integer';
3414  }
3415 
3416  if ($timelimitowner != USER_FOLDER_ID && $timelimitowner != -1) {
3417  $query .= ' AND usr_data.time_limit_owner = %s';
3418  $values[] = $timelimitowner;
3419  $types[] = 'integer';
3420  }
3421 
3422  $query .= ' ORDER BY usr_data.lastname, usr_data.firstname ';
3423 
3424  $result = $ilDB->queryF($query, $types, $values);
3425  $mem_arr = [];
3426  while ($row = $ilDB->fetchAssoc($result)) {
3427  $mem_arr[] = $row;
3428  }
3429 
3430  return $mem_arr;
3431  }
3432 
3433 
3434 
3439  public static function _getUserData(array $a_internalids): array
3440  {
3441  global $DIC;
3442 
3443  $ilDB = $DIC['ilDB'];
3444 
3445  $ids = [];
3446  if (is_array($a_internalids)) {
3447  foreach ($a_internalids as $internalid) {
3448  if (is_numeric($internalid)) {
3449  $ids[] = $internalid;
3450  } else {
3451  $parsedid = ilUtil::__extractId($internalid, IL_INST_ID);
3452  if (is_numeric($parsedid) && $parsedid > 0) {
3453  $ids[] = $parsedid;
3454  }
3455  }
3456  }
3457  }
3458  if (count($ids) == 0) {
3459  $ids [] = -1;
3460  }
3461 
3462  $query = 'SELECT usr_data.*, usr_pref.value AS language
3463  FROM usr_data
3464  LEFT JOIN usr_pref
3465  ON usr_pref.usr_id = usr_data.usr_id AND usr_pref.keyword = %s
3466  WHERE ' . $ilDB->in('usr_data.usr_id', $ids, false, 'integer');
3467  $values[] = 'language';
3468  $types[] = 'text';
3469 
3470  $query .= ' ORDER BY usr_data.lastname, usr_data.firstname ';
3471 
3472  $data = [];
3473  $result = $ilDB->queryF($query, $types, $values);
3474  while ($row = $ilDB->fetchAssoc($result)) {
3475  $data[] = $row;
3476  }
3477  return $data;
3478  }
3479 
3486  public static function _getPreferences(int $user_id): array
3487  {
3488  global $DIC;
3489 
3490  $ilDB = $DIC['ilDB'];
3491 
3492  $prefs = [];
3493 
3494  $r = $ilDB->queryF(
3495  'SELECT * FROM usr_pref WHERE usr_id = %s',
3496  ['integer'],
3497  [$user_id]
3498  );
3499 
3500  while ($row = $ilDB->fetchAssoc($r)) {
3501  $prefs[$row['keyword']] = $row['value'];
3502  }
3503 
3504  return $prefs;
3505  }
3506 
3511  public static function getUserSubsetByPreferenceValue(
3512  array $a_user_ids,
3513  string $a_keyword,
3514  string $a_val
3515  ): array {
3516  global $DIC;
3517 
3518  $ilDB = $DIC['ilDB'];
3519 
3520  $users = [];
3521  $set = $ilDB->query(
3522  'SELECT usr_id FROM usr_pref ' .
3523  ' WHERE keyword = ' . $ilDB->quote($a_keyword, 'text') .
3524  ' AND ' . $ilDB->in('usr_id', $a_user_ids, false, 'integer') .
3525  ' AND value = ' . $ilDB->quote($a_val, 'text')
3526  );
3527  while ($rec = $ilDB->fetchAssoc($set)) {
3528  $users[] = $rec['usr_id'];
3529  }
3530  return $users;
3531  }
3532 
3533  public static function _getLoginAttempts(
3534  int $a_usr_id
3535  ): int {
3536  global $DIC;
3537 
3538  $ilDB = $DIC['ilDB'];
3539 
3540  $query = 'SELECT login_attempts FROM usr_data WHERE usr_id = %s';
3541  $result = $ilDB->queryF($query, ['integer'], [$a_usr_id]);
3542  $record = $ilDB->fetchAssoc($result);
3543  return (int) ($record['login_attempts'] ?? 0);
3544  }
3545 
3546  public static function _incrementLoginAttempts(
3547  int $a_usr_id
3548  ): bool {
3549  global $DIC;
3550 
3551  $ilDB = $DIC['ilDB'];
3552 
3553  $query = 'UPDATE usr_data SET login_attempts = (login_attempts + 1) WHERE usr_id = %s';
3554  $affected = $ilDB->manipulateF($query, ['integer'], [$a_usr_id]);
3555 
3556  if ($affected) {
3557  return true;
3558  } else {
3559  return false;
3560  }
3561  }
3562 
3563  public static function _setUserInactive(
3564  int $a_usr_id
3565  ): bool {
3566  global $DIC;
3567 
3568  $ilDB = $DIC['ilDB'];
3569 
3570  $query = 'UPDATE usr_data SET active = 0, inactivation_date = %s WHERE usr_id = %s';
3571  $affected = $ilDB->manipulateF($query, ['timestamp', 'integer'], [ilUtil::now(), $a_usr_id]);
3572 
3573  if ($affected) {
3574  return true;
3575  } else {
3576  return false;
3577  }
3578  }
3579 
3583  public function hasPublicProfile(): bool
3584  {
3585  return in_array($this->getPref('public_profile'), ['y', 'g']);
3586  }
3587 
3591  public function getPublicName(): string
3592  {
3593  if ($this->hasPublicProfile()) {
3594  return $this->getFirstname() . ' ' . $this->getLastname() . ' (' . $this->getLogin() . ')';
3595  } else {
3596  return $this->getLogin();
3597  }
3598  }
3599 
3600  public static function _writeHistory(
3601  int $a_usr_id,
3602  string $a_login
3603  ): void {
3604  global $DIC;
3605 
3606  $ilDB = $DIC['ilDB'];
3607 
3608  $timestamp = time();
3609 
3610  $res = $ilDB->queryF(
3611  'SELECT * FROM loginname_history WHERE usr_id = %s AND login = %s AND history_date = %s',
3612  ['integer', 'text', 'integer'],
3613  [$a_usr_id, $a_login, $timestamp]
3614  );
3615 
3616  if ($ilDB->numRows($res) == 0) {
3617  $ilDB->manipulateF(
3618  '
3619  INSERT INTO loginname_history
3620  (usr_id, login, history_date)
3621  VALUES (%s, %s, %s)',
3622  ['integer', 'text', 'integer'],
3623  [$a_usr_id, $a_login, $timestamp]
3624  );
3625  }
3626  }
3627 
3632  public static function _getUsersOnline(
3633  int $a_user_id = 0,
3634  bool $a_no_anonymous = false
3635  ): array {
3636  global $DIC;
3637 
3638  $ilDB = $DIC->database();
3639 
3640  $log = ilLoggerFactory::getLogger('user');
3641 
3642  $pd_set = new ilSetting('pd');
3643  $atime = $pd_set->get('user_activity_time') * 60;
3644  $ctime = time();
3645 
3646  $where = [];
3647 
3648  if ($a_user_id === 0) {
3649  $where[] = 'user_id > 0';
3650  } else {
3651  $where[] = 'user_id = ' . $ilDB->quote($a_user_id, 'integer');
3652  }
3653 
3654  if ($a_no_anonymous) {
3655  $where[] = 'user_id != ' . $ilDB->quote(ANONYMOUS_USER_ID, 'integer');
3656  }
3657 
3658  if (ilUserAccountSettings::getInstance()->isUserAccessRestricted()) {
3659  $where[] = $ilDB->in('time_limit_owner', ilUserFilter::getInstance()->getFolderIds(), false, 'integer');
3660  }
3661 
3662  $where[] = 'expires > ' . $ilDB->quote($ctime, 'integer');
3663  $where[] = '(p.value IS NULL OR NOT p.value = ' . $ilDB->quote('y', 'text') . ')';
3664 
3665  $where = 'WHERE ' . implode(' AND ', $where);
3666 
3667  $r = $ilDB->queryF(
3668  $q = "
3669  SELECT COUNT(user_id) num, user_id, firstname, lastname, title, login, last_login, MAX(ctime) ctime, context, agree_date
3670  FROM usr_session
3671  LEFT JOIN usr_data u
3672  ON user_id = u.usr_id
3673  LEFT JOIN usr_pref p
3674  ON (p.usr_id = u.usr_id AND p.keyword = %s)
3675  {$where}
3676  GROUP BY user_id, firstname, lastname, title, login, last_login, context, agree_date
3677  ORDER BY lastname, firstname
3678  ",
3679  ['text'],
3680  ['hide_own_online_status']
3681  );
3682 
3683  $log->debug('Query: ' . $q);
3684 
3685  $users = [];
3686  while ($user = $ilDB->fetchAssoc($r)) {
3687  if ($atime <= 0 || $user['ctime'] + $atime > $ctime) {
3688  $users[$user['user_id']] = $user;
3689  }
3690  }
3691 
3692  $log->debug('Found users: ' . count($users));
3693 
3694  $hide_users = $DIC['legalDocuments']->usersWithHiddenOnlineStatus(array_map(intval(...), array_column($users, 'user_id')));
3695  $users = array_filter(
3696  $users,
3697  fn($user) => !in_array((int) $user['user_id'], $hide_users, true)
3698  );
3699 
3700  return $users;
3701  }
3702 
3707  public static function _generateRegistrationHash(int $a_usr_id): string
3708  {
3709  global $DIC;
3710 
3711  $ilDB = $DIC['ilDB'];
3712 
3713  do {
3714  $continue = false;
3715 
3716  $hashcode = substr(md5(uniqid(mt_rand(), true)), 0, 16);
3717 
3718  $res = $ilDB->queryf(
3719  '
3720  SELECT COUNT(usr_id) cnt FROM usr_data
3721  WHERE reg_hash = %s',
3722  ['text'],
3723  [$hashcode]
3724  );
3725  while ($row = $res->fetchRow(ilDBConstants::FETCHMODE_OBJECT)) {
3726  if ($row->cnt > 0) {
3727  $continue = true;
3728  }
3729  break;
3730  }
3731 
3732  if ($continue) {
3733  continue;
3734  }
3735 
3736  $ilDB->manipulateF(
3737  '
3738  UPDATE usr_data
3739  SET reg_hash = %s
3740  WHERE usr_id = %s',
3741  ['text', 'integer'],
3742  [$hashcode, $a_usr_id]
3743  );
3744 
3745  break;
3746  } while (true);
3747 
3748  return $hashcode;
3749  }
3750 
3756  public static function _verifyRegistrationHash(
3757  string $a_hash
3758  ): int {
3759  global $DIC;
3760 
3761  $ilDB = $DIC['ilDB'];
3762 
3763  $res = $ilDB->queryf(
3764  '
3765  SELECT usr_id, create_date FROM usr_data
3766  WHERE reg_hash = %s',
3767  ['text'],
3768  [$a_hash]
3769  );
3770  while ($row = $ilDB->fetchAssoc($res)) {
3771  $oRegSettigs = new ilRegistrationSettings();
3772 
3773  if ($oRegSettigs->getRegistrationHashLifetime() != 0 &&
3774  time() - $oRegSettigs->getRegistrationHashLifetime() > strtotime($row['create_date'])) {
3776  'reg_confirmation_hash_life_time_expired',
3777  (int) $row['usr_id']
3778  );
3779  }
3780 
3781  $ilDB->manipulateF(
3782  '
3783  UPDATE usr_data
3784  SET reg_hash = %s
3785  WHERE usr_id = %s',
3786  ['text', 'integer'],
3787  ['', (int) $row['usr_id']]
3788  );
3789 
3790  return (int) $row['usr_id'];
3791  }
3792 
3793  throw new ilRegistrationHashNotFoundException('reg_confirmation_hash_not_found');
3794  }
3795 
3796  public function setBirthday(?string $a_birthday): void
3797  {
3798  if ($a_birthday && strlen($a_birthday)) {
3799  $date = new ilDate($a_birthday, IL_CAL_DATE);
3800  $this->birthday = $date->get(IL_CAL_DATE);
3801  } else {
3802  $this->birthday = null;
3803  }
3804  }
3805 
3806  public function getBirthday(): ?string
3807  {
3808  return $this->birthday;
3809  }
3810 
3817  public static function getUserIdsByInactivityPeriod(
3818  int $periodInDays
3819  ): array {
3820  global $DIC;
3821 
3822  if ($periodInDays < 1) {
3823  throw new ilException('Invalid period given');
3824  }
3825 
3826  $date = date('Y-m-d H:i:s', (time() - ($periodInDays * 24 * 60 * 60)));
3827 
3828  $query = 'SELECT usr_id FROM usr_data WHERE last_login IS NOT NULL AND last_login < %s';
3829 
3830  $ids = [];
3831 
3832  $types = ['timestamp'];
3833  $values = [$date];
3834 
3835  $res = $DIC->database()->queryF($query, $types, $values);
3836  while ($row = $DIC->database()->fetchAssoc($res)) {
3837  $ids[] = (int) $row['usr_id'];
3838  }
3839 
3840  return $ids;
3841  }
3842 
3848  public static function getUserIdsNeverLoggedIn(
3849  int $thresholdInDays
3850  ): array {
3851  global $DIC;
3852 
3853  $date = date('Y-m-d H:i:s', (time() - ($thresholdInDays * 24 * 60 * 60)));
3854 
3855  $query = 'SELECT usr_id FROM usr_data WHERE last_login IS NULL AND create_date < %s';
3856 
3857  $ids = [];
3858 
3859  $types = ['timestamp'];
3860  $values = [$date];
3861 
3862  $res = $DIC->database()->queryF($query, $types, $values);
3863  while ($row = $DIC->database()->fetchAssoc($res)) {
3864  $ids[] = (int) $row['usr_id'];
3865  }
3866 
3867  return $ids;
3868  }
3869 
3876  public static function _getUserIdsByInactivationPeriod(
3877  int $period
3878  ): array {
3880  $field = 'inactivation_date';
3882 
3883  if (!$period) {
3884  throw new ilException('no valid period given');
3885  }
3886 
3887  global $DIC;
3888 
3889  $ilDB = $DIC['ilDB'];
3890 
3891  $date = date('Y-m-d H:i:s', (time() - ($period * 24 * 60 * 60)));
3892 
3893  $query = "SELECT usr_id FROM usr_data WHERE $field < %s AND active = %s";
3894 
3895  $res = $ilDB->queryF($query, ['timestamp', 'integer'], [$date, 0]);
3896 
3897  $ids = [];
3898  while ($row = $res->fetchRow(ilDBConstants::FETCHMODE_OBJECT)) {
3899  $ids[] = (int) $row->usr_id;
3900  }
3901 
3902  return $ids;
3903  }
3904 
3905  public function resetOwner(): void
3906  {
3907  $ilDB = $this->db;
3908 
3909  $query = 'UPDATE object_data SET owner = 0 ' .
3910  'WHERE owner = ' . $ilDB->quote($this->getId(), 'integer');
3911  $ilDB->query($query);
3912  }
3913 
3918  public static function getFirstLettersOfLastnames(
3919  ?array $user_ids = null
3920  ): array {
3921  global $DIC;
3922 
3923  $ilDB = $DIC->database();
3924 
3925  $q = 'SELECT DISTINCT ' . $ilDB->upper($ilDB->substr('lastname', 1, 1)) . ' let' .
3926  ' FROM usr_data' .
3927  ' WHERE usr_id <> ' . $ilDB->quote(ANONYMOUS_USER_ID, 'integer') .
3928  ($user_ids !== null ? ' AND ' . $ilDB->in('usr_id', $user_ids, false, 'integer') : '') .
3929  ' ORDER BY let';
3930  $let_set = $ilDB->query($q);
3931 
3932  $let = [];
3933  while ($let_rec = $ilDB->fetchAssoc($let_set)) {
3934  $let[$let_rec['let']] = $let_rec['let'];
3935  }
3936  return $let;
3937  }
3938 
3939  public static function userExists(
3940  array $a_usr_ids = []
3941  ): bool {
3942  global $DIC;
3943 
3944  $ilDB = $DIC['ilDB'];
3945 
3946  $query = 'SELECT count(*) num FROM object_data od ' .
3947  'JOIN usr_data ud ON obj_id = usr_id ' .
3948  'WHERE ' . $ilDB->in('obj_id', $a_usr_ids, false, 'integer') . ' ';
3949  $res = $ilDB->query($query);
3950  $num_rows = $res->fetchRow(ilDBConstants::FETCHMODE_OBJECT)->num;
3951  return $num_rows == count($a_usr_ids);
3952  }
3953 
3954  public function exportPersonalData(): void
3955  {
3956  $exp = new ilExport();
3957  $dir = ilExport::_getExportDirectory($this->getId(), 'xml', 'usr', 'personal_data');
3958  ilFileUtils::delDir($dir, true);
3959  $title = $this->getLastname() . ', ' . $this->getLastname() . ' [' . $this->getLogin() . ']';
3960  $exp->exportEntity(
3961  'personal_data',
3962  $this->getId(),
3963  '',
3964  'components/ILIAS/User',
3965  $title,
3966  $dir
3967  );
3968  }
3969 
3970  public function getPersonalDataExportFile(): string
3971  {
3972  $dir = ilExport::_getExportDirectory($this->getId(), 'xml', 'usr', 'personal_data');
3973  if (!is_dir($dir)) {
3974  return '';
3975  }
3976  foreach (ilFileUtils::getDir($dir) as $entry) {
3977  if (is_int(strpos($entry['entry'], '.zip'))) {
3978  return $entry['entry'];
3979  }
3980  }
3981 
3982  return '';
3983  }
3984 
3985  public function sendPersonalDataFile(): void
3986  {
3987  $file = ilExport::_getExportDirectory($this->getId(), 'xml', 'usr', 'personal_data') .
3988  '/' . $this->getPersonalDataExportFile();
3989  if (is_file($file)) {
3991  }
3992  }
3993 
3994  public function importPersonalData(
3995  array $a_file,
3996  bool $a_profile_data,
3997  bool $a_settings,
3998  bool $a_notes,
3999  bool $a_calendar
4000  ): void {
4001  $imp = new ilImport();
4002  // bookmarks need to be skipped, importer does not exist anymore
4003  $imp->addSkipImporter('components/ILIAS/Bookmarks');
4004  if (!$a_profile_data) {
4005  $imp->addSkipEntity('components/ILIAS/User', 'usr_profile');
4006  }
4007  if (!$a_settings) {
4008  $imp->addSkipEntity('components/ILIAS/User', 'usr_setting');
4009  }
4010  if (!$a_notes) {
4011  $imp->addSkipEntity('components/ILIAS/Notes', 'user_notes');
4012  }
4013  if (!$a_calendar) {
4014  $imp->addSkipEntity('components/ILIAS/Calendar', 'calendar');
4015  }
4016  $imp->importEntity(
4017  $a_file['tmp_name'],
4018  $a_file['name'],
4019  'personal_data',
4020  'components/ILIAS/User'
4021  );
4022  }
4023 
4024  public function setInactivationDate(?string $inactivation_date): void
4025  {
4026  $this->inactivation_date = $inactivation_date;
4027  }
4028 
4029  public function getInactivationDate(): ?string
4030  {
4031  return $this->inactivation_date;
4032  }
4033 
4034  public function isAnonymous(): bool
4035  {
4036  return self::_isAnonymous($this->getId());
4037  }
4038 
4039  public static function _isAnonymous(int $usr_id): bool
4040  {
4041  return $usr_id == ANONYMOUS_USER_ID;
4042  }
4043 
4044  public function activateDeletionFlag(): void
4045  {
4046  $this->writePref('delete_flag', true);
4047  }
4048 
4049  public function removeDeletionFlag(): void
4050  {
4051  $this->writePref('delete_flag', false);
4052  }
4053 
4054  public function hasDeletionFlag(): bool
4055  {
4056  return (bool) $this->getPref('delete_flag');
4057  }
4058 
4059  public function setIsSelfRegistered(bool $status): void
4060  {
4061  $this->is_self_registered = $status;
4062  }
4063 
4064  public function isSelfRegistered(): bool
4065  {
4067  }
4068 
4069 
4070  //
4071  // MULTI-TEXT / INTERESTS
4072  //
4073 
4077  public function setGeneralInterests(?array $value = null): void
4078  {
4079  $this->interests_general = $value ?? [];
4080  }
4081 
4085  public function getGeneralInterests(): array
4086  {
4087  return $this->interests_general;
4088  }
4089 
4093  public function getGeneralInterestsAsText(): string
4094  {
4095  return $this->buildTextFromArray($this->interests_general);
4096  }
4097 
4101  public function setOfferingHelp(?array $value = null): void
4102  {
4103  $this->interests_help_offered = $value ?? [];
4104  }
4105 
4109  public function getOfferingHelp(): array
4110  {
4112  }
4113 
4117  public function getOfferingHelpAsText(): string
4118  {
4119  return $this->buildTextFromArray($this->interests_help_offered);
4120  }
4121 
4125  public function setLookingForHelp(?array $value = null): void
4126  {
4127  $this->interests_help_looking = $value ?? [];
4128  }
4129 
4133  public function getLookingForHelp(): array
4134  {
4136  }
4137 
4141  public function getLookingForHelpAsText(): string
4142  {
4143  return $this->buildTextFromArray($this->interests_help_looking);
4144  }
4145 
4151  protected function buildTextFromArray(array $a_attr): string
4152  {
4153  if (count($a_attr) > 0) {
4154  return implode(', ', $a_attr);
4155  }
4156  return '';
4157  }
4158 
4159  protected function readMultiTextFields(): void
4160  {
4161  global $DIC;
4162 
4163  $ilDB = $DIC['ilDB'];
4164 
4165  if (!$this->getId()) {
4166  return;
4167  }
4168 
4169  $set = $ilDB->query('SELECT field_id,value' .
4170  ' FROM usr_data_multi' .
4171  ' WHERE usr_id = ' . $ilDB->quote($this->getId(), 'integer') .
4172  ' ORDER BY value');
4173  while ($row = $ilDB->fetchAssoc($set)) {
4174  $values[$row['field_id']][] = $row['value'];
4175  }
4176 
4177  if (isset($values['interests_general'])) {
4178  $this->setGeneralInterests($values['interests_general']);
4179  } else {
4180  $this->setGeneralInterests();
4181  }
4182  if (isset($values['interests_help_offered'])) {
4183  $this->setOfferingHelp($values['interests_help_offered']);
4184  } else {
4185  $this->setOfferingHelp();
4186  }
4187  if (isset($values['interests_help_looking'])) {
4188  $this->setLookingForHelp($values['interests_help_looking']);
4189  } else {
4190  $this->setLookingForHelp();
4191  }
4192  }
4193 
4194  public function updateMultiTextFields(bool $a_create = false): void
4195  {
4196  global $DIC;
4197 
4198  $ilDB = $DIC['ilDB'];
4199 
4200  if (!$this->getId()) {
4201  return;
4202  }
4203 
4204  if (!$a_create) {
4205  $this->deleteMultiTextFields();
4206  }
4207 
4208  $map = [
4209  'interests_general' => $this->getGeneralInterests(),
4210  'interests_help_offered' => $this->getOfferingHelp(),
4211  'interests_help_looking' => $this->getLookingForHelp()
4212  ];
4213 
4214  foreach ($map as $id => $values) {
4215  if (is_array($values) && count($values)) {
4216  foreach ($values as $value) {
4217  $value = trim($value);
4218  if ($value) {
4219  $uniq_id = $ilDB->nextId('usr_data_multi');
4220 
4221  $ilDB->manipulate('INSERT usr_data_multi' .
4222  ' (id,usr_id,field_id,value) VALUES' .
4223  ' (' . $ilDB->quote($uniq_id, 'integer') .
4224  ',' . $ilDB->quote($this->getId(), 'integer') .
4225  ',' . $ilDB->quote($id, 'text') .
4226  ',' . $ilDB->quote($value, 'text') .
4227  ')');
4228  }
4229  }
4230  }
4231  }
4232  }
4233 
4234  protected function deleteMultiTextFields(): void
4235  {
4236  global $DIC;
4237 
4238  $ilDB = $DIC['ilDB'];
4239 
4240  if (!$this->getId()) {
4241  return;
4242  }
4243 
4244  $ilDB->manipulate('DELETE FROM usr_data_multi' .
4245  ' WHERE usr_id = ' . $ilDB->quote($this->getId(), 'integer'));
4246  }
4247 
4248  public static function findInterests(
4249  string $a_term,
4250  ?int $a_user_id = null,
4251  ?string $a_field_id = null
4252  ): array {
4253  global $DIC;
4254 
4255  $ilDB = $DIC['ilDB'];
4256 
4257  $res = [];
4258 
4259  $sql = 'SELECT DISTINCT(value)' .
4260  ' FROM usr_data_multi' .
4261  ' WHERE ' . $ilDB->like('value', 'text', '%' . $a_term . '%');
4262  if ($a_field_id) {
4263  $sql .= ' AND field_id = ' . $ilDB->quote($a_field_id, 'text');
4264  }
4265  if ($a_user_id) {
4266  $sql .= ' AND usr_id <> ' . $ilDB->quote($a_user_id, 'integer');
4267  }
4268  $sql .= ' ORDER BY value';
4269  $set = $ilDB->query($sql);
4270  while ($row = $ilDB->fetchAssoc($set)) {
4271  $res[] = $row['value'];
4272  }
4273 
4274  return $res;
4275  }
4276 
4285  public static function getProfileStatusOfUsers(
4286  array $a_user_ids
4287  ): array {
4288  global $DIC;
4289 
4290  $ilDB = $DIC->database();
4291 
4292  $set = $ilDB->query(
4293  'SELECT * FROM usr_pref ' .
4294  ' WHERE keyword = ' . $ilDB->quote('public_profile', 'text') .
4295  ' AND ' . $ilDB->in('usr_id', $a_user_ids, false, 'integer')
4296  );
4297  $r = [
4298  'global' => [],
4299  'local' => [],
4300  'public' => [],
4301  'not_public' => []
4302  ];
4303  while ($rec = $ilDB->fetchAssoc($set)) {
4304  if ($rec['value'] == 'g') {
4305  $r['global'][] = $rec['usr_id'];
4306  $r['public'][] = $rec['usr_id'];
4307  }
4308  if ($rec['value'] == 'y') {
4309  $r['local'][] = $rec['usr_id'];
4310  $r['public'][] = $rec['usr_id'];
4311  }
4312  }
4313  foreach ($a_user_ids as $id) {
4314  if (!in_array($id, $r['public'])) {
4315  $r['not_public'][] = $id;
4316  }
4317  }
4318 
4319  return $r;
4320  }
4321 }
getCurrentLanguage()
returns the current language (may differ from user&#39;s pref setting!)
static getWebspaceDir(string $mode="filesystem")
get webspace directory
static hasActiveSession(int $a_user_id, string $a_session_id)
setUTitle(string $a_str)
set user title (note: don&#39;t mix up this method with setTitle() that is derived from ilObject and sets...
string $title
setProfileIncomplete(bool $a_prof_inc)
string $passwd
string $loc_zoom
static get(string $a_var)
setTimeLimitUnlimited(bool $a_unlimited)
static array static setUseRelativeDates(bool $a_status)
set use relative dates
static _setFeedPass(int $a_user_id, string $a_password)
Set news feed password for user.
setClientIP(string $a_str)
setAvatarRid(?string $avatar_rid)
string $avatar_rid
$res
Definition: ltiservices.php:66
getStoredActive(int $a_id)
get user active state
setLastLogin(string $a_str)
static _getUsersForRole(int $role_id, int $active=-1)
return array of complete users which belong to a specific role
resetPassword(string $raw, string $raw_retype)
Resets the user password.
refreshLogin()
updates the login data of a &#39;user&#39;
static _getLoginAttempts(int $a_usr_id)
static _destroyByUserId(int $a_user_id)
Destroy session.
static copyProfilePicturesToDirectory(int $a_user_id, string $a_dir)
static _deleteUser(int $a_usr_id)
getPersonalPicturePath(string $a_size='small', bool $a_force_pic=false)
static _lookupLanguage(int $a_usr_id)
buildTextFromArray(array $a_attr)
Convert multi-text values to plain text.
string $utitle
const IL_INST_ID
Definition: constants.php:40
const IL_CAL_DATETIME
const ANONYMOUS_USER_ID
Definition: constants.php:27
static getLogger(string $a_component_id)
Get component logger.
static _getAllUserData(?array $a_fields=null, int $active=-1)
const USER_FOLDER_ID
Definition: constants.php:33
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static _getAuthMode(?string $a_auth_mode)
writePref(string $a_keyword, string $a_value)
txt(string $a_topic, string $a_default_lang_fallback_mod="")
gets the text for a given topic if the topic is not in the list, the topic itself with "-" will be re...
setAgreeDate(?string $a_str)
array $interests_help_looking
static styleExists(string $style_id)
static getUserIdsByInactivityPeriod(int $periodInDays)
Get ids of all users that have been inactive for at least the given period.
string $street
static lookupMatriculation(int $a_usr_id)
static getFirstLettersOfLastnames(?array $user_ids=null)
Get first letters of all lastnames.
string $fullname
__construct(int $a_user_id=0, bool $a_call_by_reference=false)
static getProfileStatusOfUsers(array $a_user_ids)
Get profile status.
static _verifyRegistrationHash(string $a_hash)
Verifies a registration hash.
assignData(array $a_data)
loads a record &#39;user&#39; from array
fetchAssoc(ilDBStatement $statement)
static _externalAccountExists(string $a_external_account, string $a_auth_mode)
Check if an external account name already exists.
setMatriculation(string $a_str)
static _lookupFullname(int $a_user_id)
getGeneralInterestsAsText()
Get general interests as plain text.
getDefaultLanguage()
Return default language.
Class for user related exception handling in ILIAS.
setPhoneOffice(string $a_str)
addToPCClipboard(string $a_content, string $a_time, int $a_nr)
Add a page content item to PC clipboard (should go to another class)
static stripSlashes(string $a_str, bool $a_strip_html=true, string $a_allow="")
static _moveUsersToStyle(string $a_from_skin, string $a_from_style, string $a_to_skin, string $a_to_style)
static _getPreferences(int $user_id)
get preferences for user
removeObjectFromClipboard(int $a_item_id, string $a_type)
getPasswordPolicyResetStatus()
static _removeTrackingDataForUser(int $user_id)
to be called from IlObjUser
const SYSTEM_USER_ID
This file contains constants for PHPStan analyis, see: https://phpstan.org/config-reference#constants...
Definition: constants.php:26
updateLogin(string $a_login)
update login name
setSelectedCountry(string $a_val)
Set selected country (selection drop down)
deletePref(string $a_keyword)
static _lookupLastLogin(int $a_user_id)
array $interests_help_offered
string $phone_mobile
int $time_limit_from
string $time_limit_message
static array $personal_image_cache
setFullname()
builds a string with title + firstname + lastname method is used to build fullname in member variable...
const PASSWD_PLAIN
ilTree $tree
string $matriculation
static _writeAuthMode(int $a_usr_id, string $a_auth_mode)
static _lookupName(int $a_user_id)
lookup user name
static resetToDefaults()
reset to defaults
Import class.
static _lookupId($a_user_str)
static lookupTitle(int $a_page_id)
static _deleteSettingsOfUser(int $a_user)
getFullname(int $a_max_strlen=0)
string $passwd_type
removeUserPicture(bool $a_do_update=true)
static getUserIdsByEmail(string $a_email)
int $time_limit_until
static _lookupPref(int $a_usr_id, string $a_keyword)
addObjectToClipboard(int $a_item_id, string $a_type, string $a_title, int $a_parent=0, string $a_time='', int $a_order_nr=0)
add an item to user&#39;s personal clipboard
loadLanguageModule(string $a_module)
static _checkExternalAuthAccount(string $a_auth, string $a_account, bool $tryFallback=true)
check whether external account and authentication method matches with a user
static getUserSubsetByPreferenceValue(array $a_user_ids, string $a_keyword, string $a_val)
For a given set of user IDs return a subset that has a given user preference set. ...
string $longitude
getCreateDate()
Get create date in YYYY-MM-DD HH-MM-SS format.
static _lookupExternalAccount(int $a_user_id)
static _getUsersForGroup(array $a_mem_ids, int $active=-1)
return user data for group members
static _deleteByUser(int $a_usr_id)
static _needsExternalAccountByAuthMode($a_auth_mode)
Check if chosen auth mode needs an external account entry.
string $institution
const IL_CAL_UNIX
static setLanguage(ilLanguage $a_lng)
static _lookupAuthMode(int $a_usr_id)
login()
description: > Example for rendring a login glyph.
Definition: login.php:41
setPasswordSalt(?string $password_salt)
static lookupOrgUnitsRepresentation(int $a_usr_id)
lookup org unit representation
static _lookupGender(int $a_user_id)
setTimeLimitUntil(?int $a_until)
static _getAuthModeName($a_auth_key)
static _uploadPersonalPicture(string $tmp_file, int $obj_id)
Create a personal picture image file from a temporary image file.
setLoginAttempts(int $a_login_attempts)
static now()
Return current timestamp in Y-m-d H:i:s format.
static _deleteAllPref(int $a_user_id)
Deletes a userpref value of the user from the database.
static _getExportDirectory(int $a_obj_id, string $a_type="xml", string $a_obj_type="", string $a_entity="")
Get export directory for an repository object
string $latitude
setInstitution(string $a_str)
setCity(string $a_str)
getPublicName()
returns firstname lastname and login if profile is public, login otherwise
deleteUserDefinedFieldEntries()
Services $irss
getAuthMode(bool $a_auth_key=false)
static _deleteUser(int $a_usr_id)
getOfferingHelpAsText()
Get help offering as plain text.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
$ilErr
Definition: raiseError.php:33
setLatitude(?string $a_latitude)
setComment(string $a_str)
setPasswd(string $a_str, string $a_type=ilObjUser::PASSWD_PLAIN)
setGender(string $a_str)
int $last_password_change_ts
setId(int $id)
static _getAvatar(int $a_usr_id)
Factory for Date Formats.
Definition: Factory.php:26
setExternalAccount(string $a_str)
static _getNumberOfUsersForStyle(string $a_skin, string $a_style)
static deliverFileLegacy(string $a_file, ?string $a_filename=null, ?string $a_mime=null, ?bool $isInline=false, ?bool $removeAfterDelivery=false, ?bool $a_exit_after=true)
string $agree_date
setPasswordPolicyResetStatus(bool $status)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
array $user_defined_data
setBirthday(?string $a_birthday)
bool $profile_incomplete
string $department
setSkin(string $a_str)
static _writeExternalAccount(int $a_usr_id, string $a_ext_id)
string $password_salt
static _getInstance()
Get singleton instance of this class.
string $last_login
bool $time_limit_unlimited
static _getExternalAccountsByAuthMode(string $a_auth_mode, bool $a_read_auth_default=false)
Get list of external account by authentication method Note: If login == ext_account for two user with...
static getUserIdByLogin(string $a_login)
writeAccepted()
write accept date of user agreement
static _getLastHistoryDataByUserId(int $a_usr_id)
Returns the last used loginname and the changedate of the passed user_id.
ilAppEventHandler $app_event_handler
string $auth_mode
setLogin(string $a_str)
setActive(bool $a_active, int $a_owner=0)
set user active state and updates system fields appropriately
const NO_AVATAR_RID
txt(string $a_topic, string $a_default_lang_fallback_mod="")
string $phone_home
getClipboardObjects(string $a_type='', bool $a_top_nodes_only=false)
get all clipboard objects of user and specified type
static userExists(array $a_usr_ids=[])
static _lookupClientIP(int $a_user_id)
setDepartment(string $a_str)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static _lookupTitle(int $obj_id)
static deleteEntriesOfUser(int $a_user_id)
$GLOBALS["DIC"]
Definition: wac.php:53
string $last_update
setOfferingHelp(?array $value=null)
static getLoginFromAuth()
Gets the username from $ilAuth, and converts it into an ILIAS login name.
clipboardDeleteObjectsOfType(string $a_type)
static getUserLoginsByEmail(string $a_email)
getLoginByUserId(int $a_userid)
This describes how a letter or a picture avatar could be modified during construction of UI...
Definition: Avatar.php:28
A Date Format provides a format definition akin to PHP&#39;s date formatting options, but stores the sing...
Definition: DateFormat.php:26
static _getFeedPass(int $a_user_id)
Lookup news feed password for user.
static _exists(string $a_parent_type, int $a_id, string $a_lang="", bool $a_no_cache=false)
Checks whether page exists.
static _generateRegistrationHash(int $a_usr_id)
Generates a unique hashcode for activating a user profile after registration.
static styleExistsForSkinId(string $skin_id, string $style_id)
setApproveDate(?string $a_str)
set date the user account was activated null indicates that the user has not yet been activated ...
Mail Box class Base class for creating and handling mail boxes.
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static _setUserInactive(int $a_usr_id)
ilLanguage $lng
setPhoneHome(string $a_str)
static _getNumberOfUsersPerAuthMode()
get number of users per auth mode
static _toggleActiveStatusOfUsers(array $a_usr_ids, bool $a_status)
static _loginExists(string $a_login, int $a_user_id=0)
check if a login name already exists You may exclude a user from the check by giving his user id as 2...
static removeForUser(int $user_id)
Remove all notifications for given user.
getPref(string $a_keyword)
static delDir(string $a_dir, bool $a_clean_only=false)
removes a dir and all its content (subdirs and files) recursively
ilDBInterface $db
setInactivationDate(?string $inactivation_date)
syncActive()
synchronizes current and stored user active values for the owner value to be set correctly, this function should only be called when an admin is approving a user account
static _getUsersOnline(int $a_user_id=0, bool $a_no_anonymous=false)
reads all active sessions from db and returns users that are online OR returns only one active user i...
string $second_email
global $DIC
Definition: shib_login.php:22
string $birthday
updateOwner()
update owner of object in db
static _deletePref(int $a_user_id, string $a_keyword)
static _getUserData(array $a_internalids)
return user data for given user ids
checkUserId()
check user id with login name
Class ilObjForumAdministration.
string $country
static _writeHistory(int $a_usr_id, string $a_login)
getLookingForHelpAsText()
Get help looking for as plain text.
setHobby(string $a_str)
setIsSelfRegistered(bool $status)
static getDir(string $a_dir, bool $a_rec=false, ?string $a_sub_dir="")
get directory
string $phone_office
static _lookupFields(int $a_user_id)
lookup fields (deprecated; use more specific methods instead)
bool $passwd_policy_reset
setPref(string $a_keyword, ?string $a_value)
Class ilUserAvatarResolver.
const UDF_TYPE_WYSIWYG
static _getUserIdsByInactivationPeriod(int $period)
get ids of all users that have been inactivated since at least the given period
setFirstLogin(string $a_str)
string $gender
setCountry(string $a_str)
setEmail(string $a_str)
setFax(string $a_str)
queryF(string $query, array $types, array $values)
getPCClipboardContent()
Add a page content item to PC clipboard (should go to another class)
static _isAnonymous(int $usr_id)
existsPref(string $a_keyword)
importPersonalData(array $a_file, bool $a_profile_data, bool $a_settings, bool $a_notes, bool $a_calendar)
const PASSWD_CRYPTED
foreach($mandatory_scripts as $file) $timestamp
Definition: buildRTE.php:70
int $time_limit_owner
setPhoneMobile(string $a_str)
string $first_login
getSelectedCountry()
Get selected country (selection drop down)
static deleteByUserId(int $a_user_id)
setLastPasswordChangeTS(int $a_last_password_change_ts)
setLastname(string $a_str)
setPasswordEncodingType(?string $password_encryption_type)
static _getPersonalPicturePath(int $a_usr_id, string $a_size='small', bool $a_force_pic=false, bool $a_prevent_no_photo_image=false, bool $html_export=false)
setStreet(string $a_str)
string $lastname
static _doesLoginnameExistInHistory(string $a_login)
Checks whether the passed loginname already exists in history.
static _getUsersForClipboadObject(string $a_type, int $a_id)
get all users, that have a certain object within their clipboard
static deleteUserPortfolios(int $a_user_id)
Delete all portfolio data for user.
Class ilObjAuthSettingsGUI.
bool $is_self_registered
static findInterests(string $a_term, ?int $a_user_id=null, ?string $a_field_id=null)
static skinExists(string $skin_id, ?ilSystemStyleConfig $system_style_config=null)
Check whether a skin exists.
const IL_CAL_DATE
string $ext_account
static _incrementLoginAttempts(int $a_usr_id)
string $last_profile_prompt
static _lookupFeedHash(int $a_user_id, bool $a_create=false)
Lookup news feed hash for user.
setLastProfilePrompt(string $a_str)
static _getImportedUserId(string $i2_id)
static _getUsersForFolder(int $ref_id, int $active)
get users for a category or from system folder
setLanguage(string $a_str)
static _readUsersProfileData(array $a_user_ids)
global $ilSetting
Definition: privfeed.php:31
static _lookup(int $a_user_id, string $a_field)
static __extractId(string $ilias_id, int $inst_id)
extract ref id from role title, e.g.
string $org_units
$id
plugin.php for ilComponentBuildPluginInfoObjectiveTest::testAddPlugins
Definition: plugin.php:23
static toUsernameWithoutDomain(string $a_login)
Static function removes Microsoft domain name from username webdav related.
__construct(Container $dic, ilPlugin $plugin)
setTimeLimitOwner(int $a_owner)
updateMultiTextFields(bool $a_create=false)
static formatDate(ilDateTime $date, bool $a_skip_day=false, bool $a_include_wd=false, bool $include_seconds=false, ?ilObjUser $user=null,)
$q
Definition: shib_logout.php:21
setSecondEmail(?string $second_email)
string $password_encoding_type
static _writePref(int $a_usr_id, string $a_keyword, string $a_value)
setLongitude(?string $a_longitude)
array $interests_general
setLocationZoom(?int $a_locationzoom)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
ilLogger $log
DateFormatFactory $date_format_factory
setLastPasswordChangeToNow()
string $approve_date
setZipcode(string $a_str)
ilCronDeleteInactiveUserReminderMail $cron_delete_user_reminder_mail
setTimeLimitMessage(string $a_time_limit_message)
static _lookupFirstLogin(int $a_user_id)
debug(string $message, array $context=[])
Class for user related exception handling in ILIAS.
string $client_ip
setUserDefinedData(array $a_data)
static _lookupType(int $id, bool $reference=false)
string $zipcode
setGeneralInterests(?array $value=null)
setTimeLimitFrom(?int $a_from)
getClipboardChilds(int $a_parent, string $a_insert_time)
Get children of an item.
string $referral_comment
setFirstname(string $a_str)
string $inactivation_date
static _lookupEmail(int $a_user_id)
string $sel_country
static clear(string $a_var)
static _getInstance()
Get instance of ilSecuritySettings.
string $firstname
static set(string $a_var, $a_val)
Set a value.
static _getUsersForIds(array $a_mem_ids, int $active=-1, int $timelimitowner=-1)
return user data for given user id
setAuthMode(?string $a_str)
setCurrentLanguage(string $a_val)
Set current language.
static _lookupActive(int $a_usr_id)
hasPublicProfile()
returns true if public is profile, false otherwise
static _getAllUserAssignedStyles()
setLastUpdate(string $a_str)
static getUserIdsNeverLoggedIn(int $thresholdInDays)
Get ids of all users that have never logged in.
setOwner(int $usr_id)
static _removeTrackingDataForUser(int $user_id)
static formatPeriod(ilDateTime $start, ilDateTime $end, bool $a_skip_starting_day=false, ?ilObjUser $user=null)
Format a period of two dates Shows: 14.
static _getLocalAccountsForEmail(string $a_email)
clipboardHasObjectsOfType(string $a_type)
Check whether clipboard has objects of a certain type.
ilObjUser $user
$r
static _lookupLogin(int $a_user_id)
setLookingForHelp(?array $value=null)