See Release Notes
Long Term Support Release
Differences Between: [Versions 39 and 310] [Versions 39 and 311] [Versions 39 and 400] [Versions 39 and 401] [Versions 39 and 402] [Versions 39 and 403]
1 <?php 2 // This file is part of Moodle - http://moodle.org/ 3 // 4 // Moodle is free software: you can redistribute it and/or modify 5 // it under the terms of the GNU General Public License as published by 6 // the Free Software Foundation, either version 3 of the License, or 7 // (at your option) any later version. 8 // 9 // Moodle is distributed in the hope that it will be useful, 10 // but WITHOUT ANY WARRANTY; without even the implied warranty of 11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 // GNU General Public License for more details. 13 // 14 // You should have received a copy of the GNU General Public License 15 // along with Moodle. If not, see <http://www.gnu.org/licenses/>. 16 17 /** 18 * User class 19 * 20 * @package core 21 * @copyright 2013 Rajesh Taneja <rajesh@moodle.com> 22 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later 23 */ 24 25 defined('MOODLE_INTERNAL') || die(); 26 27 /** 28 * User class to access user details. 29 * 30 * @todo move api's from user/lib.php and deprecate old ones. 31 * @package core 32 * @copyright 2013 Rajesh Taneja <rajesh@moodle.com> 33 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later 34 */ 35 class core_user { 36 /** 37 * No reply user id. 38 */ 39 const NOREPLY_USER = -10; 40 41 /** 42 * Support user id. 43 */ 44 const SUPPORT_USER = -20; 45 46 /** 47 * Hide email address from everyone. 48 */ 49 const MAILDISPLAY_HIDE = 0; 50 51 /** 52 * Display email address to everyone. 53 */ 54 const MAILDISPLAY_EVERYONE = 1; 55 56 /** 57 * Display email address to course members only. 58 */ 59 const MAILDISPLAY_COURSE_MEMBERS_ONLY = 2; 60 61 /** 62 * List of fields that can be synched/locked during authentication. 63 */ 64 const AUTHSYNCFIELDS = [ 65 'firstname', 66 'lastname', 67 'email', 68 'city', 69 'country', 70 'lang', 71 'description', 72 'url', 73 'idnumber', 74 'institution', 75 'department', 76 'phone1', 77 'phone2', 78 'address', 79 'firstnamephonetic', 80 'lastnamephonetic', 81 'middlename', 82 'alternatename' 83 ]; 84 85 /** @var int Indicates that user profile view should be prevented */ 86 const VIEWPROFILE_PREVENT = -1; 87 /** @var int Indicates that user profile view should not be prevented */ 88 const VIEWPROFILE_DO_NOT_PREVENT = 0; 89 /** @var int Indicates that user profile view should be allowed even if Moodle would prevent it */ 90 const VIEWPROFILE_FORCE_ALLOW = 1; 91 92 /** @var stdClass keep record of noreply user */ 93 public static $noreplyuser = false; 94 95 /** @var stdClass keep record of support user */ 96 public static $supportuser = false; 97 98 /** @var array store user fields properties cache. */ 99 protected static $propertiescache = null; 100 101 /** @var array store user preferences cache. */ 102 protected static $preferencescache = null; 103 104 /** 105 * Return user object from db or create noreply or support user, 106 * if userid matches corse_user::NOREPLY_USER or corse_user::SUPPORT_USER 107 * respectively. If userid is not found, then return false. 108 * 109 * @param int $userid user id 110 * @param string $fields A comma separated list of user fields to be returned, support and noreply user 111 * will not be filtered by this. 112 * @param int $strictness IGNORE_MISSING means compatible mode, false returned if user not found, debug message if more found; 113 * IGNORE_MULTIPLE means return first user, ignore multiple user records found(not recommended); 114 * MUST_EXIST means throw an exception if no user record or multiple records found. 115 * @return stdClass|bool user record if found, else false. 116 * @throws dml_exception if user record not found and respective $strictness is set. 117 */ 118 public static function get_user($userid, $fields = '*', $strictness = IGNORE_MISSING) { 119 global $DB; 120 121 // If noreply user then create fake record and return. 122 switch ($userid) { 123 case self::NOREPLY_USER: 124 return self::get_noreply_user(); 125 break; 126 case self::SUPPORT_USER: 127 return self::get_support_user(); 128 break; 129 default: 130 return $DB->get_record('user', array('id' => $userid), $fields, $strictness); 131 } 132 } 133 134 /** 135 * Return user object from db based on their email. 136 * 137 * @param string $email The email of the user searched. 138 * @param string $fields A comma separated list of user fields to be returned, support and noreply user. 139 * @param int $mnethostid The id of the remote host. 140 * @param int $strictness IGNORE_MISSING means compatible mode, false returned if user not found, debug message if more found; 141 * IGNORE_MULTIPLE means return first user, ignore multiple user records found(not recommended); 142 * MUST_EXIST means throw an exception if no user record or multiple records found. 143 * @return stdClass|bool user record if found, else false. 144 * @throws dml_exception if user record not found and respective $strictness is set. 145 */ 146 public static function get_user_by_email($email, $fields = '*', $mnethostid = null, $strictness = IGNORE_MISSING) { 147 global $DB, $CFG; 148 149 // Because we use the username as the search criteria, we must also restrict our search based on mnet host. 150 if (empty($mnethostid)) { 151 // If empty, we restrict to local users. 152 $mnethostid = $CFG->mnet_localhost_id; 153 } 154 155 return $DB->get_record('user', array('email' => $email, 'mnethostid' => $mnethostid), $fields, $strictness); 156 } 157 158 /** 159 * Return user object from db based on their username. 160 * 161 * @param string $username The username of the user searched. 162 * @param string $fields A comma separated list of user fields to be returned, support and noreply user. 163 * @param int $mnethostid The id of the remote host. 164 * @param int $strictness IGNORE_MISSING means compatible mode, false returned if user not found, debug message if more found; 165 * IGNORE_MULTIPLE means return first user, ignore multiple user records found(not recommended); 166 * MUST_EXIST means throw an exception if no user record or multiple records found. 167 * @return stdClass|bool user record if found, else false. 168 * @throws dml_exception if user record not found and respective $strictness is set. 169 */ 170 public static function get_user_by_username($username, $fields = '*', $mnethostid = null, $strictness = IGNORE_MISSING) { 171 global $DB, $CFG; 172 173 // Because we use the username as the search criteria, we must also restrict our search based on mnet host. 174 if (empty($mnethostid)) { 175 // If empty, we restrict to local users. 176 $mnethostid = $CFG->mnet_localhost_id; 177 } 178 179 return $DB->get_record('user', array('username' => $username, 'mnethostid' => $mnethostid), $fields, $strictness); 180 } 181 182 /** 183 * Searches for users by name, possibly within a specified context, with current user's access. 184 * 185 * Deciding which users to search is complicated because it relies on user permissions; 186 * ideally, we shouldn't show names if you aren't allowed to see their profile. The permissions 187 * for seeing profile are really complicated. 188 * 189 * Even if search is restricted to a course, it's possible that other people might have 190 * been able to contribute within the course (e.g. they were enrolled before and not now; 191 * or people with system-level roles) so if the user has permission we do want to include 192 * everyone. However, if there are multiple results then we prioritise the ones who are 193 * enrolled in the course. 194 * 195 * If you have moodle/user:viewdetails at system level, you can search everyone. 196 * Otherwise we check which courses you *do* have that permission and search everyone who is 197 * enrolled on those courses. 198 * 199 * Normally you can only search the user's name. If you have the moodle/site:viewuseridentity 200 * capability then we also let you search the fields which are listed as identity fields in 201 * the 'showuseridentity' config option. For example, this might include the user's ID number 202 * or email. 203 * 204 * The $max parameter controls the maximum number of users returned. If users are restricted 205 * from view for some reason, multiple runs of the main query might be made; the $querylimit 206 * parameter allows this to be restricted. Both parameters can be zero to remove limits. 207 * 208 * The returned user objects include id, username, all fields required for user pictures, and 209 * user identity fields. 210 * 211 * @param string $query Search query text 212 * @param \context_course|null $coursecontext Course context or null if system-wide 213 * @param int $max Max number of users to return, default 30 (zero = no limit) 214 * @param int $querylimit Max number of database queries, default 5 (zero = no limit) 215 * @return array Array of user objects with limited fields 216 */ 217 public static function search($query, \context_course $coursecontext = null, 218 $max = 30, $querylimit = 5) { 219 global $CFG, $DB; 220 require_once($CFG->dirroot . '/user/lib.php'); 221 222 // Allow limits to be turned off. 223 if (!$max) { 224 $max = PHP_INT_MAX; 225 } 226 if (!$querylimit) { 227 $querylimit = PHP_INT_MAX; 228 } 229 230 // Check permission to view profiles at each context. 231 $systemcontext = \context_system::instance(); 232 $viewsystem = has_capability('moodle/user:viewdetails', $systemcontext); 233 if ($viewsystem) { 234 $userquery = 'SELECT id FROM {user}'; 235 $userparams = []; 236 } 237 if (!$viewsystem) { 238 list($userquery, $userparams) = self::get_enrolled_sql_on_courses_with_capability( 239 'moodle/user:viewdetails'); 240 if (!$userquery) { 241 // No permissions anywhere, return nothing. 242 return []; 243 } 244 } 245 246 // Start building the WHERE clause based on name. 247 list ($where, $whereparams) = users_search_sql($query, 'u', false); 248 249 // We allow users to search with extra identity fields (as well as name) but only if they 250 // have the permission to display those identity fields. 251 $extrasql = ''; 252 $extraparams = []; 253 254 if (empty($CFG->showuseridentity)) { 255 // Explode gives wrong result with empty string. 256 $extra = []; 257 } else { 258 $extra = explode(',', $CFG->showuseridentity); 259 } 260 261 // We need the username just to skip guests. 262 $extrafieldlist = $extra; 263 if (!in_array('username', $extra)) { 264 $extrafieldlist[] = 'username'; 265 } 266 // The deleted flag will always be false because users_search_sql excludes deleted users, 267 // but it must be present or it causes PHP warnings in some functions below. 268 if (!in_array('deleted', $extra)) { 269 $extrafieldlist[] = 'deleted'; 270 } 271 $selectfields = \user_picture::fields('u', 272 array_merge(get_all_user_name_fields(), $extrafieldlist)); 273 274 $index = 1; 275 foreach ($extra as $fieldname) { 276 if ($extrasql) { 277 $extrasql .= ' OR '; 278 } 279 $extrasql .= $DB->sql_like('u.' . $fieldname, ':extra' . $index, false); 280 $extraparams['extra' . $index] = $query . '%'; 281 $index++; 282 } 283 284 $identitysystem = has_capability('moodle/site:viewuseridentity', $systemcontext); 285 $usingshowidentity = false; 286 if ($identitysystem) { 287 // They have permission everywhere so just add the extra query to the normal query. 288 $where .= ' OR ' . $extrasql; 289 $whereparams = array_merge($whereparams, $extraparams); 290 } else { 291 // Get all courses where user can view full user identity. 292 list($sql, $params) = self::get_enrolled_sql_on_courses_with_capability( 293 'moodle/site:viewuseridentity'); 294 if ($sql) { 295 // Join that with the user query to get an extra field indicating if we can. 296 $userquery = " 297 SELECT innerusers.id, COUNT(identityusers.id) AS showidentity 298 FROM ($userquery) innerusers 299 LEFT JOIN ($sql) identityusers ON identityusers.id = innerusers.id 300 GROUP BY innerusers.id"; 301 $userparams = array_merge($userparams, $params); 302 $usingshowidentity = true; 303 304 // Query on the extra fields only in those places. 305 $where .= ' OR (users.showidentity > 0 AND (' . $extrasql . '))'; 306 $whereparams = array_merge($whereparams, $extraparams); 307 } 308 } 309 310 // Default order is just name order. But if searching within a course then we show users 311 // within the course first. 312 list ($order, $orderparams) = users_order_by_sql('u', $query, $systemcontext); 313 if ($coursecontext) { 314 list ($sql, $params) = get_enrolled_sql($coursecontext); 315 $mainfield = 'innerusers2.id'; 316 if ($usingshowidentity) { 317 $mainfield .= ', innerusers2.showidentity'; 318 } 319 $userquery = " 320 SELECT $mainfield, COUNT(courseusers.id) AS incourse 321 FROM ($userquery) innerusers2 322 LEFT JOIN ($sql) courseusers ON courseusers.id = innerusers2.id 323 GROUP BY $mainfield"; 324 $userparams = array_merge($userparams, $params); 325 326 $order = 'incourse DESC, ' . $order; 327 } 328 329 // Get result (first 30 rows only) from database. Take a couple spare in case we have to 330 // drop some. 331 $result = []; 332 $got = 0; 333 $pos = 0; 334 $readcount = $max + 2; 335 for ($i = 0; $i < $querylimit; $i++) { 336 $rawresult = $DB->get_records_sql(" 337 SELECT $selectfields 338 FROM ($userquery) users 339 JOIN {user} u ON u.id = users.id 340 WHERE $where 341 ORDER BY $order", array_merge($userparams, $whereparams, $orderparams), 342 $pos, $readcount); 343 foreach ($rawresult as $user) { 344 // Skip guest. 345 if ($user->username === 'guest') { 346 continue; 347 } 348 // Check user can really view profile (there are per-user cases where this could 349 // be different for some reason, this is the same check used by the profile view pages 350 // to double-check that it is OK). 351 if (!user_can_view_profile($user)) { 352 continue; 353 } 354 $result[] = $user; 355 $got++; 356 if ($got >= $max) { 357 break; 358 } 359 } 360 361 if ($got >= $max) { 362 // All necessary results obtained. 363 break; 364 } 365 if (count($rawresult) < $readcount) { 366 // No more results from database. 367 break; 368 } 369 $pos += $readcount; 370 } 371 372 return $result; 373 } 374 375 /** 376 * Gets an SQL query that lists all enrolled user ids on any course where the current 377 * user has the specified capability. Helper function used for searching users. 378 * 379 * @param string $capability Required capability 380 * @return array Array containing SQL and params, or two nulls if there are no courses 381 */ 382 protected static function get_enrolled_sql_on_courses_with_capability($capability) { 383 // Get all courses where user have the capability. 384 $courses = get_user_capability_course($capability, null, true, 385 implode(',', array_values(context_helper::get_preload_record_columns('ctx')))); 386 if (!$courses) { 387 return [null, null]; 388 } 389 390 // Loop around all courses getting the SQL for enrolled users. Note: This query could 391 // probably be more efficient (without the union) if get_enrolled_sql had a way to 392 // pass an array of courseids, but it doesn't. 393 $unionsql = ''; 394 $unionparams = []; 395 foreach ($courses as $course) { 396 // Get SQL to list user ids enrolled in this course. 397 \context_helper::preload_from_record($course); 398 list ($sql, $params) = get_enrolled_sql(\context_course::instance($course->id)); 399 400 // Combine to a big union query. 401 if ($unionsql) { 402 $unionsql .= ' UNION '; 403 } 404 $unionsql .= $sql; 405 $unionparams = array_merge($unionparams, $params); 406 } 407 408 return [$unionsql, $unionparams]; 409 } 410 411 /** 412 * Helper function to return dummy noreply user record. 413 * 414 * @return stdClass 415 */ 416 protected static function get_dummy_user_record() { 417 global $CFG; 418 419 $dummyuser = new stdClass(); 420 $dummyuser->id = self::NOREPLY_USER; 421 $dummyuser->email = $CFG->noreplyaddress; 422 $dummyuser->firstname = get_string('noreplyname'); 423 $dummyuser->username = 'noreply'; 424 $dummyuser->lastname = ''; 425 $dummyuser->confirmed = 1; 426 $dummyuser->suspended = 0; 427 $dummyuser->deleted = 0; 428 $dummyuser->picture = 0; 429 $dummyuser->auth = 'manual'; 430 $dummyuser->firstnamephonetic = ''; 431 $dummyuser->lastnamephonetic = ''; 432 $dummyuser->middlename = ''; 433 $dummyuser->alternatename = ''; 434 $dummyuser->imagealt = ''; 435 return $dummyuser; 436 } 437 438 /** 439 * Return noreply user record, this is currently used in messaging 440 * system only for sending messages from noreply email. 441 * It will return record of $CFG->noreplyuserid if set else return dummy 442 * user object with hard-coded $user->emailstop = 1 so noreply can be sent to user. 443 * 444 * @return stdClass user record. 445 */ 446 public static function get_noreply_user() { 447 global $CFG; 448 449 if (!empty(self::$noreplyuser)) { 450 return self::$noreplyuser; 451 } 452 453 // If noreply user is set then use it, else create one. 454 if (!empty($CFG->noreplyuserid)) { 455 self::$noreplyuser = self::get_user($CFG->noreplyuserid); 456 self::$noreplyuser->emailstop = 1; // Force msg stop for this user. 457 return self::$noreplyuser; 458 } else { 459 // Do not cache the dummy user record to avoid language internationalization issues. 460 $noreplyuser = self::get_dummy_user_record(); 461 $noreplyuser->maildisplay = '1'; // Show to all. 462 $noreplyuser->emailstop = 1; 463 return $noreplyuser; 464 } 465 } 466 467 /** 468 * Return support user record, this is currently used in messaging 469 * system only for sending messages to support email. 470 * $CFG->supportuserid is set then returns user record 471 * $CFG->supportemail is set then return dummy record with $CFG->supportemail 472 * else return admin user record with hard-coded $user->emailstop = 0, so user 473 * gets support message. 474 * 475 * @return stdClass user record. 476 */ 477 public static function get_support_user() { 478 global $CFG; 479 480 if (!empty(self::$supportuser)) { 481 return self::$supportuser; 482 } 483 484 // If custom support user is set then use it, else if supportemail is set then use it, else use noreply. 485 if (!empty($CFG->supportuserid)) { 486 self::$supportuser = self::get_user($CFG->supportuserid, '*', MUST_EXIST); 487 } else if (empty(self::$supportuser) && !empty($CFG->supportemail)) { 488 // Try sending it to support email if support user is not set. 489 $supportuser = self::get_dummy_user_record(); 490 $supportuser->id = self::SUPPORT_USER; 491 $supportuser->email = $CFG->supportemail; 492 if ($CFG->supportname) { 493 $supportuser->firstname = $CFG->supportname; 494 } 495 $supportuser->username = 'support'; 496 $supportuser->maildisplay = '1'; // Show to all. 497 // Unset emailstop to make sure support message is sent. 498 $supportuser->emailstop = 0; 499 return $supportuser; 500 } 501 502 // Send support msg to admin user if nothing is set above. 503 if (empty(self::$supportuser)) { 504 self::$supportuser = get_admin(); 505 } 506 507 // Unset emailstop to make sure support message is sent. 508 self::$supportuser->emailstop = 0; 509 return self::$supportuser; 510 } 511 512 /** 513 * Reset self::$noreplyuser and self::$supportuser. 514 * This is only used by phpunit, and there is no other use case for this function. 515 * Please don't use it outside phpunit. 516 */ 517 public static function reset_internal_users() { 518 if (PHPUNIT_TEST) { 519 self::$noreplyuser = false; 520 self::$supportuser = false; 521 } else { 522 debugging('reset_internal_users() should not be used outside phpunit.', DEBUG_DEVELOPER); 523 } 524 } 525 526 /** 527 * Return true if user id is greater than 0 and alternatively check db. 528 * 529 * @param int $userid user id. 530 * @param bool $checkdb if true userid will be checked in db. By default it's false, and 531 * userid is compared with 0 for performance. 532 * @return bool true is real user else false. 533 */ 534 public static function is_real_user($userid, $checkdb = false) { 535 global $DB; 536 537 if ($userid <= 0) { 538 return false; 539 } 540 if ($checkdb) { 541 return $DB->record_exists('user', array('id' => $userid)); 542 } else { 543 return true; 544 } 545 } 546 547 /** 548 * Check if the given user is an active user in the site. 549 * 550 * @param stdClass $user user object 551 * @param boolean $checksuspended whether to check if the user has the account suspended 552 * @param boolean $checknologin whether to check if the user uses the nologin auth method 553 * @throws moodle_exception 554 * @since Moodle 3.0 555 */ 556 public static function require_active_user($user, $checksuspended = false, $checknologin = false) { 557 558 if (!self::is_real_user($user->id)) { 559 throw new moodle_exception('invaliduser', 'error'); 560 } 561 562 if ($user->deleted) { 563 throw new moodle_exception('userdeleted'); 564 } 565 566 if (empty($user->confirmed)) { 567 throw new moodle_exception('usernotconfirmed', 'moodle', '', $user->username); 568 } 569 570 if (isguestuser($user)) { 571 throw new moodle_exception('guestsarenotallowed', 'error'); 572 } 573 574 if ($checksuspended and $user->suspended) { 575 throw new moodle_exception('suspended', 'auth'); 576 } 577 578 if ($checknologin and $user->auth == 'nologin') { 579 throw new moodle_exception('suspended', 'auth'); 580 } 581 } 582 583 /** 584 * Updates the provided users profile picture based upon the expected fields returned from the edit or edit_advanced forms. 585 * 586 * @param stdClass $usernew An object that contains some information about the user being updated 587 * @param array $filemanageroptions 588 * @return bool True if the user was updated, false if it stayed the same. 589 */ 590 public static function update_picture(stdClass $usernew, $filemanageroptions = array()) { 591 global $CFG, $DB; 592 require_once("$CFG->libdir/gdlib.php"); 593 594 $context = context_user::instance($usernew->id, MUST_EXIST); 595 $user = core_user::get_user($usernew->id, 'id, picture', MUST_EXIST); 596 597 $newpicture = $user->picture; 598 // Get file_storage to process files. 599 $fs = get_file_storage(); 600 if (!empty($usernew->deletepicture)) { 601 // The user has chosen to delete the selected users picture. 602 $fs->delete_area_files($context->id, 'user', 'icon'); // Drop all images in area. 603 $newpicture = 0; 604 } 605 606 // Save newly uploaded file, this will avoid context mismatch for newly created users. 607 if (!isset($usernew->imagefile)) { 608 $usernew->imagefile = 0; 609 } 610 file_save_draft_area_files($usernew->imagefile, $context->id, 'user', 'newicon', 0, $filemanageroptions); 611 if (($iconfiles = $fs->get_area_files($context->id, 'user', 'newicon')) && count($iconfiles) == 2) { 612 // Get file which was uploaded in draft area. 613 foreach ($iconfiles as $file) { 614 if (!$file->is_directory()) { 615 break; 616 } 617 } 618 // Copy file to temporary location and the send it for processing icon. 619 if ($iconfile = $file->copy_content_to_temp()) { 620 // There is a new image that has been uploaded. 621 // Process the new image and set the user to make use of it. 622 // NOTE: Uploaded images always take over Gravatar. 623 $newpicture = (int)process_new_icon($context, 'user', 'icon', 0, $iconfile); 624 // Delete temporary file. 625 @unlink($iconfile); 626 // Remove uploaded file. 627 $fs->delete_area_files($context->id, 'user', 'newicon'); 628 } else { 629 // Something went wrong while creating temp file. 630 // Remove uploaded file. 631 $fs->delete_area_files($context->id, 'user', 'newicon'); 632 return false; 633 } 634 } 635 636 if ($newpicture != $user->picture) { 637 $DB->set_field('user', 'picture', $newpicture, array('id' => $user->id)); 638 return true; 639 } else { 640 return false; 641 } 642 } 643 644 645 646 /** 647 * Definition of user profile fields and the expected parameter type for data validation. 648 * 649 * array( 650 * 'property_name' => array( // The user property to be checked. Should match the field on the user table. 651 * 'null' => NULL_ALLOWED, // Defaults to NULL_NOT_ALLOWED. Takes NULL_NOT_ALLOWED or NULL_ALLOWED. 652 * 'type' => PARAM_TYPE, // Expected parameter type of the user field. 653 * 'choices' => array(1, 2..) // An array of accepted values of the user field. 654 * 'default' => $CFG->setting // An default value for the field. 655 * ) 656 * ) 657 * 658 * The fields choices and default are optional. 659 * 660 * @return void 661 */ 662 protected static function fill_properties_cache() { 663 global $CFG; 664 if (self::$propertiescache !== null) { 665 return; 666 } 667 668 // Array of user fields properties and expected parameters. 669 // Every new field on the user table should be added here otherwise it won't be validated. 670 $fields = array(); 671 $fields['id'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED); 672 $fields['auth'] = array('type' => PARAM_AUTH, 'null' => NULL_NOT_ALLOWED); 673 $fields['confirmed'] = array('type' => PARAM_BOOL, 'null' => NULL_NOT_ALLOWED); 674 $fields['policyagreed'] = array('type' => PARAM_BOOL, 'null' => NULL_NOT_ALLOWED); 675 $fields['deleted'] = array('type' => PARAM_BOOL, 'null' => NULL_NOT_ALLOWED); 676 $fields['suspended'] = array('type' => PARAM_BOOL, 'null' => NULL_NOT_ALLOWED); 677 $fields['mnethostid'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED); 678 $fields['username'] = array('type' => PARAM_USERNAME, 'null' => NULL_NOT_ALLOWED); 679 $fields['password'] = array('type' => PARAM_RAW, 'null' => NULL_NOT_ALLOWED); 680 $fields['idnumber'] = array('type' => PARAM_RAW, 'null' => NULL_NOT_ALLOWED); 681 $fields['firstname'] = array('type' => PARAM_NOTAGS, 'null' => NULL_NOT_ALLOWED); 682 $fields['lastname'] = array('type' => PARAM_NOTAGS, 'null' => NULL_NOT_ALLOWED); 683 $fields['surname'] = array('type' => PARAM_NOTAGS, 'null' => NULL_NOT_ALLOWED); 684 $fields['email'] = array('type' => PARAM_RAW_TRIMMED, 'null' => NULL_NOT_ALLOWED); 685 $fields['emailstop'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 'default' => 0); 686 $fields['icq'] = array('type' => PARAM_NOTAGS, 'null' => NULL_NOT_ALLOWED); 687 $fields['skype'] = array('type' => PARAM_NOTAGS, 'null' => NULL_ALLOWED); 688 $fields['aim'] = array('type' => PARAM_NOTAGS, 'null' => NULL_NOT_ALLOWED); 689 $fields['yahoo'] = array('type' => PARAM_NOTAGS, 'null' => NULL_NOT_ALLOWED); 690 $fields['msn'] = array('type' => PARAM_NOTAGS, 'null' => NULL_NOT_ALLOWED); 691 $fields['phone1'] = array('type' => PARAM_NOTAGS, 'null' => NULL_NOT_ALLOWED); 692 $fields['phone2'] = array('type' => PARAM_NOTAGS, 'null' => NULL_NOT_ALLOWED); 693 $fields['institution'] = array('type' => PARAM_TEXT, 'null' => NULL_NOT_ALLOWED); 694 $fields['department'] = array('type' => PARAM_TEXT, 'null' => NULL_NOT_ALLOWED); 695 $fields['address'] = array('type' => PARAM_TEXT, 'null' => NULL_NOT_ALLOWED); 696 $fields['city'] = array('type' => PARAM_TEXT, 'null' => NULL_NOT_ALLOWED, 'default' => $CFG->defaultcity); 697 $fields['country'] = array('type' => PARAM_ALPHA, 'null' => NULL_NOT_ALLOWED, 'default' => $CFG->country, 698 'choices' => array_merge(array('' => ''), get_string_manager()->get_list_of_countries(true, true))); 699 $fields['lang'] = array('type' => PARAM_LANG, 'null' => NULL_NOT_ALLOWED, 'default' => $CFG->lang, 700 'choices' => array_merge(array('' => ''), get_string_manager()->get_list_of_translations(false))); 701 $fields['calendartype'] = array('type' => PARAM_PLUGIN, 'null' => NULL_NOT_ALLOWED, 'default' => $CFG->calendartype, 702 'choices' => array_merge(array('' => ''), \core_calendar\type_factory::get_list_of_calendar_types())); 703 $fields['theme'] = array('type' => PARAM_THEME, 'null' => NULL_NOT_ALLOWED, 704 'default' => theme_config::DEFAULT_THEME, 'choices' => array_merge(array('' => ''), get_list_of_themes())); 705 $fields['timezone'] = array('type' => PARAM_TIMEZONE, 'null' => NULL_NOT_ALLOWED, 706 'default' => core_date::get_server_timezone()); // Must not use choices here: timezones can come and go. 707 $fields['firstaccess'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED); 708 $fields['lastaccess'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED); 709 $fields['lastlogin'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED); 710 $fields['currentlogin'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED); 711 $fields['lastip'] = array('type' => PARAM_NOTAGS, 'null' => NULL_NOT_ALLOWED); 712 $fields['secret'] = array('type' => PARAM_ALPHANUM, 'null' => NULL_NOT_ALLOWED); 713 $fields['picture'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED); 714 $fields['url'] = array('type' => PARAM_URL, 'null' => NULL_NOT_ALLOWED); 715 $fields['description'] = array('type' => PARAM_RAW, 'null' => NULL_ALLOWED); 716 $fields['descriptionformat'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED); 717 $fields['mailformat'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 718 'default' => $CFG->defaultpreference_mailformat); 719 $fields['maildigest'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 720 'default' => $CFG->defaultpreference_maildigest); 721 $fields['maildisplay'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 722 'default' => $CFG->defaultpreference_maildisplay); 723 $fields['autosubscribe'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 724 'default' => $CFG->defaultpreference_autosubscribe); 725 $fields['trackforums'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 726 'default' => $CFG->defaultpreference_trackforums); 727 $fields['timecreated'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED); 728 $fields['timemodified'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED); 729 $fields['trustbitmask'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED); 730 $fields['imagealt'] = array('type' => PARAM_TEXT, 'null' => NULL_ALLOWED); 731 $fields['lastnamephonetic'] = array('type' => PARAM_NOTAGS, 'null' => NULL_ALLOWED); 732 $fields['firstnamephonetic'] = array('type' => PARAM_NOTAGS, 'null' => NULL_ALLOWED); 733 $fields['middlename'] = array('type' => PARAM_NOTAGS, 'null' => NULL_ALLOWED); 734 $fields['alternatename'] = array('type' => PARAM_NOTAGS, 'null' => NULL_ALLOWED); 735 736 self::$propertiescache = $fields; 737 } 738 739 /** 740 * Get properties of a user field. 741 * 742 * @param string $property property name to be retrieved. 743 * @throws coding_exception if the requested property name is invalid. 744 * @return array the property definition. 745 */ 746 public static function get_property_definition($property) { 747 748 self::fill_properties_cache(); 749 750 if (!array_key_exists($property, self::$propertiescache)) { 751 throw new coding_exception('Invalid property requested.'); 752 } 753 754 return self::$propertiescache[$property]; 755 } 756 757 /** 758 * Validate user data. 759 * 760 * This method just validates each user field and return an array of errors. It doesn't clean the data, 761 * the methods clean() and clean_field() should be used for this purpose. 762 * 763 * @param stdClass|array $data user data object or array to be validated. 764 * @return array|true $errors array of errors found on the user object, true if the validation passed. 765 */ 766 public static function validate($data) { 767 // Get all user profile fields definition. 768 self::fill_properties_cache(); 769 770 foreach ($data as $property => $value) { 771 try { 772 if (isset(self::$propertiescache[$property])) { 773 validate_param($value, self::$propertiescache[$property]['type'], self::$propertiescache[$property]['null']); 774 } 775 // Check that the value is part of a list of allowed values. 776 if (!empty(self::$propertiescache[$property]['choices']) && 777 !isset(self::$propertiescache[$property]['choices'][$value])) { 778 throw new invalid_parameter_exception($value); 779 } 780 } catch (invalid_parameter_exception $e) { 781 $errors[$property] = $e->getMessage(); 782 } 783 } 784 785 return empty($errors) ? true : $errors; 786 } 787 788 /** 789 * Clean the properties cache. 790 * 791 * During unit tests we need to be able to reset all caches so that each new test starts in a known state. 792 * Intended for use only for testing, phpunit calls this before every test. 793 */ 794 public static function reset_caches() { 795 self::$propertiescache = null; 796 } 797 798 /** 799 * Clean the user data. 800 * 801 * @param stdClass|array $user the user data to be validated against properties definition. 802 * @return stdClass $user the cleaned user data. 803 */ 804 public static function clean_data($user) { 805 if (empty($user)) { 806 return $user; 807 } 808 809 foreach ($user as $field => $value) { 810 // Get the property parameter type and do the cleaning. 811 try { 812 $user->$field = core_user::clean_field($value, $field); 813 } catch (coding_exception $e) { 814 debugging("The property '$field' could not be cleaned.", DEBUG_DEVELOPER); 815 } 816 } 817 818 return $user; 819 } 820 821 /** 822 * Clean a specific user field. 823 * 824 * @param string $data the user field data to be cleaned. 825 * @param string $field the user field name on the property definition cache. 826 * @return string the cleaned user data. 827 */ 828 public static function clean_field($data, $field) { 829 if (empty($data) || empty($field)) { 830 return $data; 831 } 832 833 try { 834 $type = core_user::get_property_type($field); 835 836 if (isset(self::$propertiescache[$field]['choices'])) { 837 if (!array_key_exists($data, self::$propertiescache[$field]['choices'])) { 838 if (isset(self::$propertiescache[$field]['default'])) { 839 $data = self::$propertiescache[$field]['default']; 840 } else { 841 $data = ''; 842 } 843 } else { 844 return $data; 845 } 846 } else { 847 $data = clean_param($data, $type); 848 } 849 } catch (coding_exception $e) { 850 debugging("The property '$field' could not be cleaned.", DEBUG_DEVELOPER); 851 } 852 853 return $data; 854 } 855 856 /** 857 * Get the parameter type of the property. 858 * 859 * @param string $property property name to be retrieved. 860 * @throws coding_exception if the requested property name is invalid. 861 * @return int the property parameter type. 862 */ 863 public static function get_property_type($property) { 864 865 self::fill_properties_cache(); 866 867 if (!array_key_exists($property, self::$propertiescache)) { 868 throw new coding_exception('Invalid property requested: ' . $property); 869 } 870 871 return self::$propertiescache[$property]['type']; 872 } 873 874 /** 875 * Discover if the property is NULL_ALLOWED or NULL_NOT_ALLOWED. 876 * 877 * @param string $property property name to be retrieved. 878 * @throws coding_exception if the requested property name is invalid. 879 * @return bool true if the property is NULL_ALLOWED, false otherwise. 880 */ 881 public static function get_property_null($property) { 882 883 self::fill_properties_cache(); 884 885 if (!array_key_exists($property, self::$propertiescache)) { 886 throw new coding_exception('Invalid property requested: ' . $property); 887 } 888 889 return self::$propertiescache[$property]['null']; 890 } 891 892 /** 893 * Get the choices of the property. 894 * 895 * This is a helper method to validate a value against a list of acceptable choices. 896 * For instance: country, language, themes and etc. 897 * 898 * @param string $property property name to be retrieved. 899 * @throws coding_exception if the requested property name is invalid or if it does not has a list of choices. 900 * @return array the property parameter type. 901 */ 902 public static function get_property_choices($property) { 903 904 self::fill_properties_cache(); 905 906 if (!array_key_exists($property, self::$propertiescache) && !array_key_exists('choices', 907 self::$propertiescache[$property])) { 908 909 throw new coding_exception('Invalid property requested, or the property does not has a list of choices.'); 910 } 911 912 return self::$propertiescache[$property]['choices']; 913 } 914 915 /** 916 * Get the property default. 917 * 918 * This method gets the default value of a field (if exists). 919 * 920 * @param string $property property name to be retrieved. 921 * @throws coding_exception if the requested property name is invalid or if it does not has a default value. 922 * @return string the property default value. 923 */ 924 public static function get_property_default($property) { 925 926 self::fill_properties_cache(); 927 928 if (!array_key_exists($property, self::$propertiescache) || !isset(self::$propertiescache[$property]['default'])) { 929 throw new coding_exception('Invalid property requested, or the property does not has a default value.'); 930 } 931 932 return self::$propertiescache[$property]['default']; 933 } 934 935 /** 936 * Definition of updateable user preferences and rules for data and access validation. 937 * 938 * array( 939 * 'preferencename' => array( // Either exact preference name or a regular expression. 940 * 'null' => NULL_ALLOWED, // Defaults to NULL_NOT_ALLOWED. Takes NULL_NOT_ALLOWED or NULL_ALLOWED. 941 * 'type' => PARAM_TYPE, // Expected parameter type of the user field - mandatory 942 * 'choices' => array(1, 2..) // An array of accepted values of the user field - optional 943 * 'default' => $CFG->setting // An default value for the field - optional 944 * 'isregex' => false/true // Whether the name of the preference is a regular expression (default false). 945 * 'permissioncallback' => callable // Function accepting arguments ($user, $preferencename) that checks if current user 946 * // is allowed to modify this preference for given user. 947 * // If not specified core_user::default_preference_permission_check() will be assumed. 948 * 'cleancallback' => callable // Custom callback for cleaning value if something more difficult than just type/choices is needed 949 * // accepts arguments ($value, $preferencename) 950 * ) 951 * ) 952 * 953 * @return void 954 */ 955 protected static function fill_preferences_cache() { 956 if (self::$preferencescache !== null) { 957 return; 958 } 959 960 // Array of user preferences and expected types/values. 961 // Every preference that can be updated directly by user should be added here. 962 $preferences = array(); 963 $preferences['auth_forcepasswordchange'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 'choices' => array(0, 1), 964 'permissioncallback' => function($user, $preferencename) { 965 global $USER; 966 $systemcontext = context_system::instance(); 967 return ($USER->id != $user->id && (has_capability('moodle/user:update', $systemcontext) || 968 ($user->timecreated > time() - 10 && has_capability('moodle/user:create', $systemcontext)))); 969 }); 970 $preferences['usemodchooser'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 'default' => 1, 971 'choices' => array(0, 1)); 972 $preferences['forum_markasreadonnotification'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 'default' => 1, 973 'choices' => array(0, 1)); 974 $preferences['htmleditor'] = array('type' => PARAM_NOTAGS, 'null' => NULL_ALLOWED, 975 'cleancallback' => function($value, $preferencename) { 976 if (empty($value) || !array_key_exists($value, core_component::get_plugin_list('editor'))) { 977 return null; 978 } 979 return $value; 980 }); 981 $preferences['badgeprivacysetting'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 'default' => 1, 982 'choices' => array(0, 1), 'permissioncallback' => function($user, $preferencename) { 983 global $CFG, $USER; 984 return !empty($CFG->enablebadges) && $user->id == $USER->id; 985 }); 986 $preferences['blogpagesize'] = array('type' => PARAM_INT, 'null' => NULL_NOT_ALLOWED, 'default' => 10, 987 'permissioncallback' => function($user, $preferencename) { 988 global $USER; 989 return $USER->id == $user->id && has_capability('moodle/blog:view', context_system::instance()); 990 }); 991 $preferences['user_home_page_preference'] = array('type' => PARAM_INT, 'null' => NULL_ALLOWED, 'default' => HOMEPAGE_MY, 992 'choices' => array(HOMEPAGE_SITE, HOMEPAGE_MY), 993 'permissioncallback' => function ($user, $preferencename) { 994 global $CFG, $USER; 995 return $user->id == $USER->id && 996 (!empty($CFG->defaulthomepage) && ($CFG->defaulthomepage == HOMEPAGE_USER)); 997 } 998 ); 999 1000 // Core components that may want to define their preferences. 1001 // List of core components implementing callback is hardcoded here for performance reasons. 1002 // TODO MDL-58184 cache list of core components implementing a function. 1003 $corecomponents = ['core_message', 'core_calendar', 'core_contentbank']; 1004 foreach ($corecomponents as $component) { 1005 if (($pluginpreferences = component_callback($component, 'user_preferences')) && is_array($pluginpreferences)) { 1006 $preferences += $pluginpreferences; 1007 } 1008 } 1009 1010 // Plugins that may define their preferences. 1011 if ($pluginsfunction = get_plugins_with_function('user_preferences')) { 1012 foreach ($pluginsfunction as $plugintype => $plugins) { 1013 foreach ($plugins as $function) { 1014 if (($pluginpreferences = call_user_func($function)) && is_array($pluginpreferences)) { 1015 $preferences += $pluginpreferences; 1016 } 1017 } 1018 } 1019 } 1020 1021 self::$preferencescache = $preferences; 1022 } 1023 1024 /** 1025 * Retrieves the preference definition 1026 * 1027 * @param string $preferencename 1028 * @return array 1029 */ 1030 protected static function get_preference_definition($preferencename) { 1031 self::fill_preferences_cache(); 1032 1033 foreach (self::$preferencescache as $key => $preference) { 1034 if (empty($preference['isregex'])) { 1035 if ($key === $preferencename) { 1036 return $preference; 1037 } 1038 } else { 1039 if (preg_match($key, $preferencename)) { 1040 return $preference; 1041 } 1042 } 1043 } 1044 1045 throw new coding_exception('Invalid preference requested.'); 1046 } 1047 1048 /** 1049 * Default callback used for checking if current user is allowed to change permission of user $user 1050 * 1051 * @param stdClass $user 1052 * @param string $preferencename 1053 * @return bool 1054 */ 1055 protected static function default_preference_permission_check($user, $preferencename) { 1056 global $USER; 1057 if (is_mnet_remote_user($user)) { 1058 // Can't edit MNET user. 1059 return false; 1060 } 1061 1062 if ($user->id == $USER->id) { 1063 // Editing own profile. 1064 $systemcontext = context_system::instance(); 1065 return has_capability('moodle/user:editownprofile', $systemcontext); 1066 } else { 1067 // Teachers, parents, etc. 1068 $personalcontext = context_user::instance($user->id); 1069 if (!has_capability('moodle/user:editprofile', $personalcontext)) { 1070 return false; 1071 } 1072 if (is_siteadmin($user->id) and !is_siteadmin($USER)) { 1073 // Only admins may edit other admins. 1074 return false; 1075 } 1076 return true; 1077 } 1078 } 1079 1080 /** 1081 * Can current user edit preference of this/another user 1082 * 1083 * @param string $preferencename 1084 * @param stdClass $user 1085 * @return bool 1086 */ 1087 public static function can_edit_preference($preferencename, $user) { 1088 if (!isloggedin() || isguestuser()) { 1089 // Guests can not edit anything. 1090 return false; 1091 } 1092 1093 try { 1094 $definition = self::get_preference_definition($preferencename); 1095 } catch (coding_exception $e) { 1096 return false; 1097 } 1098 1099 if ($user->deleted || !context_user::instance($user->id, IGNORE_MISSING)) { 1100 // User is deleted. 1101 return false; 1102 } 1103 1104 if (isset($definition['permissioncallback'])) { 1105 $callback = $definition['permissioncallback']; 1106 if (is_callable($callback)) { 1107 return call_user_func_array($callback, [$user, $preferencename]); 1108 } else { 1109 throw new coding_exception('Permission callback for preference ' . s($preferencename) . ' is not callable'); 1110 return false; 1111 } 1112 } else { 1113 return self::default_preference_permission_check($user, $preferencename); 1114 } 1115 } 1116 1117 /** 1118 * Clean value of a user preference 1119 * 1120 * @param string $value the user preference value to be cleaned. 1121 * @param string $preferencename the user preference name 1122 * @return string the cleaned preference value 1123 */ 1124 public static function clean_preference($value, $preferencename) { 1125 1126 $definition = self::get_preference_definition($preferencename); 1127 1128 if (isset($definition['type']) && $value !== null) { 1129 $value = clean_param($value, $definition['type']); 1130 } 1131 1132 if (isset($definition['cleancallback'])) { 1133 $callback = $definition['cleancallback']; 1134 if (is_callable($callback)) { 1135 return $callback($value, $preferencename); 1136 } else { 1137 throw new coding_exception('Clean callback for preference ' . s($preferencename) . ' is not callable'); 1138 } 1139 } else if ($value === null && (!isset($definition['null']) || $definition['null'] == NULL_ALLOWED)) { 1140 return null; 1141 } else if (isset($definition['choices'])) { 1142 if (!in_array($value, $definition['choices'])) { 1143 if (isset($definition['default'])) { 1144 return $definition['default']; 1145 } else { 1146 $first = reset($definition['choices']); 1147 return $first; 1148 } 1149 } else { 1150 return $value; 1151 } 1152 } else { 1153 if ($value === null) { 1154 return isset($definition['default']) ? $definition['default'] : ''; 1155 } 1156 return $value; 1157 } 1158 } 1159 1160 }
title
Description
Body
title
Description
Body
title
Description
Body
title
Body