2009-05-22 08:41:00 +00:00
< ? php
2009-11-01 11:31:16 +00:00
// This file is part of Moodle - http://moodle.org/
//
2009-05-22 08:41:00 +00:00
// Moodle is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Moodle is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
2009-11-01 11:31:16 +00:00
//
2009-05-22 08:41:00 +00:00
// You should have received a copy of the GNU General Public License
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
2008-04-15 21:46:04 +00:00
2004-09-23 02:48:41 +00:00
/**
* Library of functions for database manipulation .
2007-08-02 23:39:28 +00:00
*
2004-09-23 02:48:41 +00:00
* Other main libraries :
* - weblib . php - functions that produce web output
* - moodlelib . php - general - purpose Moodle functions
2009-05-22 08:41:00 +00:00
*
2010-07-25 13:35:05 +00:00
* @ package core
* @ copyright 1999 onwards Martin Dougiamas { @ link http :// moodle . com }
* @ license http :// www . gnu . org / copyleft / gpl . html GNU GPL v3 or later
2004-09-23 02:48:41 +00:00
*/
2010-07-25 13:35:05 +00:00
defined ( 'MOODLE_INTERNAL' ) || die ();
2010-03-31 07:41:31 +00:00
/**
* The maximum courses in a category
* MAX_COURSES_IN_CATEGORY * MAX_COURSE_CATEGORIES must not be more than max integer !
*/
2009-05-22 08:41:00 +00:00
define ( 'MAX_COURSES_IN_CATEGORY' , 10000 );
2010-03-31 07:41:31 +00:00
2009-11-01 11:31:16 +00:00
/**
2009-05-22 08:41:00 +00:00
* The maximum number of course categories
2009-11-01 11:31:16 +00:00
* MAX_COURSES_IN_CATEGORY * MAX_COURSE_CATEGORIES must not be more than max integer !
2009-05-22 08:41:00 +00:00
*/
2008-06-16 14:25:53 +00:00
define ( 'MAX_COURSE_CATEGORIES' , 10000 );
2010-03-31 07:41:31 +00:00
/**
* Number of seconds to wait before updating lastaccess information in DB .
2013-09-20 12:06:38 +08:00
*
* We allow overwrites from config . php , useful to ensure coherence in performance
* tests results .
2019-03-25 16:31:46 +00:00
*
* Note : For web service requests in the external_tokens field , we use a different constant
* webservice :: TOKEN_LASTACCESS_UPDATE_SECS .
2010-03-31 07:41:31 +00:00
*/
2013-09-20 12:06:38 +08:00
if ( ! defined ( 'LASTACCESS_UPDATE_SECS' )) {
define ( 'LASTACCESS_UPDATE_SECS' , 60 );
}
2002-12-17 04:41:18 +00:00
2003-09-14 04:04:15 +00:00
/**
2004-09-24 21:28:22 +00:00
* Returns $user object of the main admin user
2009-05-22 08:41:00 +00:00
*
2010-12-16 08:29:23 +01:00
* @ static stdClass $mainadmin
2010-12-21 19:19:35 +01:00
* @ return stdClass { @ link $USER } record from DB , false if not found
2004-09-24 21:28:22 +00:00
*/
2010-03-31 07:41:31 +00:00
function get_admin () {
2011-12-07 11:18:10 +01:00
global $CFG , $DB ;
2010-03-31 07:41:31 +00:00
static $mainadmin = null ;
2012-10-09 19:13:29 +02:00
static $prevadmins = null ;
2007-09-19 07:22:01 +00:00
2012-10-26 10:43:39 +08:00
if ( empty ( $CFG -> siteadmins )) {
// Should not happen on an ordinary site.
// It does however happen during unit tests.
2012-10-09 19:13:29 +02:00
return false ;
2011-12-07 11:18:10 +01:00
}
2012-10-09 19:13:29 +02:00
if ( isset ( $mainadmin ) and $prevadmins === $CFG -> siteadmins ) {
return clone ( $mainadmin );
2011-12-09 09:04:43 +01:00
}
2012-10-09 19:13:29 +02:00
$mainadmin = null ;
2011-12-07 11:18:10 +01:00
foreach ( explode ( ',' , $CFG -> siteadmins ) as $id ) {
if ( $user = $DB -> get_record ( 'user' , array ( 'id' => $id , 'deleted' => 0 ))) {
$mainadmin = $user ;
break ;
2002-12-17 04:41:18 +00:00
}
}
2011-12-07 11:18:10 +01:00
if ( $mainadmin ) {
2012-10-09 19:13:29 +02:00
$prevadmins = $CFG -> siteadmins ;
2011-12-07 11:18:10 +01:00
return clone ( $mainadmin );
} else {
// this should not happen
return false ;
}
2002-12-17 04:41:18 +00:00
}
2003-09-14 04:04:15 +00:00
/**
2010-03-31 07:41:31 +00:00
* Returns list of all admins , using 1 DB query
2004-09-24 21:28:22 +00:00
*
2009-05-22 08:41:00 +00:00
* @ return array
2004-09-24 21:28:22 +00:00
*/
2002-12-17 04:41:18 +00:00
function get_admins () {
2010-03-31 07:41:31 +00:00
global $DB , $CFG ;
2007-08-02 23:39:28 +00:00
2010-05-17 05:17:12 +00:00
if ( empty ( $CFG -> siteadmins )) { // Should not happen on an ordinary site
2010-05-18 16:59:08 +00:00
return array ();
2010-05-17 05:17:12 +00:00
}
2010-03-31 07:41:31 +00:00
$sql = " SELECT u.*
2008-05-25 09:39:02 +00:00
FROM { user } u
2010-03-31 07:41:31 +00:00
WHERE u . deleted = 0 AND u . id IN ( $CFG -> siteadmins ) " ;
2007-08-02 23:39:28 +00:00
2012-10-09 19:13:29 +02:00
// We want the same order as in $CFG->siteadmins.
$records = $DB -> get_records_sql ( $sql );
$admins = array ();
foreach ( explode ( ',' , $CFG -> siteadmins ) as $id ) {
$id = ( int ) $id ;
if ( ! isset ( $records [ $id ])) {
// User does not exist, this should not happen.
continue ;
}
$admins [ $records [ $id ] -> id ] = $records [ $id ];
}
return $admins ;
2002-12-17 04:41:18 +00:00
}
2004-08-21 12:41:40 +00:00
/**
2004-09-24 21:28:22 +00:00
* Search through course users
*
2007-08-02 23:39:28 +00:00
* If $coursid specifies the site course then this function searches
2004-09-24 21:28:22 +00:00
* through all undeleted and confirmed users
*
2009-05-22 08:41:00 +00:00
* @ global object
* @ uses SITEID
* @ uses SQL_PARAMS_NAMED
* @ uses CONTEXT_COURSE
2004-09-24 21:28:22 +00:00
* @ param int $courseid The course in question .
* @ param int $groupid The group in question .
2009-05-22 08:41:00 +00:00
* @ param string $searchtext The string to search for
* @ param string $sort A field to sort by
* @ param array $exceptions A list of IDs to ignore , eg 2 , 4 , 5 , 8 , 9 , 10
* @ return array
2004-09-24 21:28:22 +00:00
*/
2008-05-25 09:39:02 +00:00
function search_users ( $courseid , $groupid , $searchtext , $sort = '' , array $exceptions = null ) {
global $DB ;
2004-01-31 03:30:31 +00:00
2008-06-09 19:48:24 +00:00
$fullname = $DB -> sql_fullname ( 'u.firstname' , 'u.lastname' );
2004-09-21 11:41:58 +00:00
2004-08-21 12:41:40 +00:00
if ( ! empty ( $exceptions )) {
2011-04-14 15:13:28 +02:00
list ( $exceptions , $params ) = $DB -> get_in_or_equal ( $exceptions , SQL_PARAMS_NAMED , 'ex' , false );
2008-05-25 09:39:02 +00:00
$except = " AND u.id $exceptions " ;
2004-08-21 12:41:40 +00:00
} else {
2008-05-25 09:39:02 +00:00
$except = " " ;
$params = array ();
2004-08-21 12:41:40 +00:00
}
2004-08-30 17:27:00 +00:00
2004-08-21 12:41:40 +00:00
if ( ! empty ( $sort )) {
2008-05-25 09:39:02 +00:00
$order = " ORDER BY $sort " ;
2004-08-21 12:41:40 +00:00
} else {
2008-05-25 09:39:02 +00:00
$order = " " ;
2004-08-21 12:41:40 +00:00
}
2004-09-21 11:41:58 +00:00
2010-09-04 12:45:47 +00:00
$select = " u.deleted = 0 AND u.confirmed = 1 AND ( " . $DB -> sql_like ( $fullname , ':search1' , false ) . " OR " . $DB -> sql_like ( 'u.email' , ':search2' , false ) . " ) " ;
2008-05-25 09:39:02 +00:00
$params [ 'search1' ] = " % $searchtext % " ;
$params [ 'search2' ] = " % $searchtext % " ;
2004-08-30 17:27:00 +00:00
2004-08-29 14:15:40 +00:00
if ( ! $courseid or $courseid == SITEID ) {
2008-05-25 09:39:02 +00:00
$sql = " SELECT u.id, u.firstname, u.lastname, u.email
FROM { user } u
WHERE $select
$except
$order " ;
return $DB -> get_records_sql ( $sql , $params );
2004-08-30 17:27:00 +00:00
2008-05-25 09:39:02 +00:00
} else {
2004-08-21 12:41:40 +00:00
if ( $groupid ) {
2008-05-25 09:39:02 +00:00
$sql = " SELECT u.id, u.firstname, u.lastname, u.email
FROM { user } u
JOIN { groups_members } gm ON gm . userid = u . id
WHERE $select AND gm . groupid = : groupid
$except
$order " ;
$params [ 'groupid' ] = $groupid ;
return $DB -> get_records_sql ( $sql , $params );
2004-08-21 12:41:40 +00:00
} else {
2012-07-25 16:25:55 +08:00
$context = context_course :: instance ( $courseid );
2013-07-09 13:34:39 +08:00
// We want to query both the current context and parent contexts.
list ( $relatedctxsql , $relatedctxparams ) = $DB -> get_in_or_equal ( $context -> get_parent_context_ids ( true ), SQL_PARAMS_NAMED , 'relatedctx' );
2008-05-25 09:39:02 +00:00
$sql = " SELECT u.id, u.firstname, u.lastname, u.email
FROM { user } u
JOIN { role_assignments } ra ON ra . userid = u . id
2013-07-09 13:34:39 +08:00
WHERE $select AND ra . contextid $relatedctxsql
2008-05-25 09:39:02 +00:00
$except
$order " ;
2013-07-09 13:34:39 +08:00
$params = array_merge ( $params , $relatedctxparams );
2008-05-25 09:39:02 +00:00
return $DB -> get_records_sql ( $sql , $params );
2004-08-21 12:41:40 +00:00
}
}
2002-12-17 04:41:18 +00:00
}
2013-03-27 11:30:22 +00:00
/**
* Returns SQL used to search through user table to find users ( in a query
* which may also join and apply other conditions ) .
*
* You can combine this SQL with an existing query by adding 'AND $sql' to the
* WHERE clause of your query ( where $sql is the first element in the array
* returned by this function ), and merging in the $params array to the parameters
* of your query ( where $params is the second element ) . Your query should use
* named parameters such as : param , rather than the question mark style .
*
* There are examples of basic usage in the unit test for this function .
*
* @ param string $search the text to search for ( empty string = find all )
* @ param string $u the table alias for the user table in the query being
* built . May be '' .
* @ param bool $searchanywhere If true ( default ), searches in the middle of
* names , otherwise only searches at start
* @ param array $extrafields Array of extra user fields to include in search
* @ param array $exclude Array of user ids to exclude ( empty = don ' t exclude )
* @ param array $includeonly If specified , only returns users that have ids
* incldued in this array ( empty = don ' t restrict )
* @ return array an array with two elements , a fragment of SQL to go in the
* where clause the query , and an associative array containing any required
* parameters ( using named placeholders ) .
*/
function users_search_sql ( $search , $u = 'u' , $searchanywhere = true , array $extrafields = array (),
2013-04-03 11:44:23 +01:00
array $exclude = null , array $includeonly = null ) {
2013-03-27 11:30:22 +00:00
global $DB , $CFG ;
$params = array ();
$tests = array ();
if ( $u ) {
$u .= '.' ;
}
// If we have a $search string, put a field LIKE '$search%' condition on each field.
if ( $search ) {
$conditions = array (
$DB -> sql_fullname ( $u . 'firstname' , $u . 'lastname' ),
$conditions [] = $u . 'lastname'
);
foreach ( $extrafields as $field ) {
$conditions [] = $u . $field ;
}
if ( $searchanywhere ) {
$searchparam = '%' . $search . '%' ;
} else {
$searchparam = $search . '%' ;
}
$i = 0 ;
foreach ( $conditions as $key => $condition ) {
$conditions [ $key ] = $DB -> sql_like ( $condition , " :con { $i } 00 " , false , false );
$params [ " con { $i } 00 " ] = $searchparam ;
$i ++ ;
}
$tests [] = '(' . implode ( ' OR ' , $conditions ) . ')' ;
}
// Add some additional sensible conditions.
$tests [] = $u . " id <> :guestid " ;
$params [ 'guestid' ] = $CFG -> siteguest ;
$tests [] = $u . 'deleted = 0' ;
$tests [] = $u . 'confirmed = 1' ;
// If we are being asked to exclude any users, do that.
if ( ! empty ( $exclude )) {
list ( $usertest , $userparams ) = $DB -> get_in_or_equal ( $exclude , SQL_PARAMS_NAMED , 'ex' , false );
$tests [] = $u . 'id ' . $usertest ;
$params = array_merge ( $params , $userparams );
}
// If we are validating a set list of userids, add an id IN (...) test.
if ( ! empty ( $includeonly )) {
list ( $usertest , $userparams ) = $DB -> get_in_or_equal ( $includeonly , SQL_PARAMS_NAMED , 'val' );
$tests [] = $u . 'id ' . $usertest ;
$params = array_merge ( $params , $userparams );
}
// In case there are no tests, add one result (this makes it easier to combine
// this with an existing query as you can always add AND $sql).
if ( empty ( $tests )) {
$tests [] = '1 = 1' ;
}
// Combing the conditions and return.
return array ( implode ( ' AND ' , $tests ), $params );
}
2012-08-08 13:53:28 +01:00
/**
* This function generates the standard ORDER BY clause for use when generating
* lists of users . If you don ' t have a reason to use a different order , then
* you should use this method to generate the order when displaying lists of users .
*
* If the optional $search parameter is passed , then exact matches to the search
* will be sorted first . For example , suppose you have two users 'Al Zebra' and
* 'Alan Aardvark' . The default sort is Alan , then Al . If , however , you search for
* 'Al' , then Al will be listed first . ( With two users , this is not a big deal ,
* but with thousands of users , it is essential . )
*
* The list of fields scanned for exact matches are :
* - firstname
* - lastname
* - $DB -> sql_fullname
* - those returned by get_extra_user_fields
*
* If named parameters are used ( which is the default , and highly recommended ),
* then the parameter names are like : usersortexactN , where N is an int .
*
* The simplest possible example use is :
* list ( $sort , $params ) = users_order_by_sql ();
* $sql = 'SELECT * FROM {users} ORDER BY ' . $sort ;
*
* A more complex example , showing that this sort can be combined with other sorts :
* list ( $sort , $sortparams ) = users_order_by_sql ( 'u' );
* $sql = " SELECT g.id AS groupid, gg.groupingid, u.id AS userid, u.firstname, u.lastname, u.idnumber, u.username
* FROM { groups } g
* LEFT JOIN { groupings_groups } gg ON g . id = gg . groupid
* LEFT JOIN { groups_members } gm ON g . id = gm . groupid
* LEFT JOIN { user } u ON gm . userid = u . id
* WHERE g . courseid = : courseid $groupwhere $groupingwhere
* ORDER BY g . name , $sort " ;
* $params += $sortparams ;
*
* An example showing the use of $search :
* list ( $sort , $sortparams ) = users_order_by_sql ( 'u' , $search , $this -> get_context ());
* $order = ' ORDER BY ' . $sort ;
* $params += $sortparams ;
* $availableusers = $DB -> get_records_sql ( $fields . $sql . $order , $params , $page * $perpage , $perpage );
*
* @ param string $usertablealias ( optional ) any table prefix for the { users } table . E . g . 'u' .
* @ param string $search ( optional ) a current search string . If given ,
* any exact matches to this string will be sorted first .
* @ param context $context the context we are in . Use by get_extra_user_fields .
* Defaults to $PAGE -> context .
* @ return array with two elements :
* string SQL fragment to use in the ORDER BY clause . For example , " firstname, lastname " .
* array of parameters used in the SQL fragment .
*/
function users_order_by_sql ( $usertablealias = '' , $search = null , context $context = null ) {
global $DB , $PAGE ;
if ( $usertablealias ) {
$tableprefix = $usertablealias . '.' ;
} else {
$tableprefix = '' ;
}
$sort = " { $tableprefix } lastname, { $tableprefix } firstname, { $tableprefix } id " ;
$params = array ();
if ( ! $search ) {
return array ( $sort , $params );
}
if ( ! $context ) {
$context = $PAGE -> context ;
}
$exactconditions = array ();
$paramkey = 'usersortexact1' ;
$exactconditions [] = $DB -> sql_fullname ( $tableprefix . 'firstname' , $tableprefix . 'lastname' ) .
' = :' . $paramkey ;
$params [ $paramkey ] = $search ;
$paramkey ++ ;
$fieldstocheck = array_merge ( array ( 'firstname' , 'lastname' ), get_extra_user_fields ( $context ));
foreach ( $fieldstocheck as $key => $field ) {
2012-10-04 11:43:33 +01:00
$exactconditions [] = 'LOWER(' . $tableprefix . $field . ') = LOWER(:' . $paramkey . ')' ;
2012-08-08 13:53:28 +01:00
$params [ $paramkey ] = $search ;
$paramkey ++ ;
}
$sort = 'CASE WHEN ' . implode ( ' OR ' , $exactconditions ) .
' THEN 0 ELSE 1 END, ' . $sort ;
return array ( $sort , $params );
}
2003-09-14 04:04:15 +00:00
/**
2004-09-24 21:28:22 +00:00
* Returns a subset of users
*
2009-05-22 08:41:00 +00:00
* @ global object
* @ uses DEBUG_DEVELOPER
* @ uses SQL_PARAMS_NAMED
2005-07-12 02:23:58 +00:00
* @ param bool $get If false then only a count of the records is returned
2004-09-24 21:28:22 +00:00
* @ param string $search A simple string to search for
2005-07-12 02:23:58 +00:00
* @ param bool $confirmed A switch to allow / disallow unconfirmed users
2009-05-22 08:41:00 +00:00
* @ param array $exceptions A list of IDs to ignore , eg 2 , 4 , 5 , 8 , 9 , 10
2004-09-24 21:28:22 +00:00
* @ param string $sort A SQL snippet for the sorting criteria to use
2009-05-22 08:41:00 +00:00
* @ param string $firstinitial Users whose first name starts with $firstinitial
* @ param string $lastinitial Users whose last name starts with $lastinitial
* @ param string $page The page or records to return
* @ param string $recordsperpage The number of records to return per page
2004-09-24 21:28:22 +00:00
* @ param string $fields A comma separated list of fields to be returned from the chosen table .
2009-11-01 11:31:16 +00:00
* @ return array | int | bool { @ link $USER } records unless get is false in which case the integer count of the records found is returned .
2012-08-08 13:53:28 +01:00
* False is returned if an error is encountered .
2004-09-24 21:28:22 +00:00
*/
2008-05-25 09:39:02 +00:00
function get_users ( $get = true , $search = '' , $confirmed = false , array $exceptions = null , $sort = 'firstname ASC' ,
$firstinitial = '' , $lastinitial = '' , $page = '' , $recordsperpage = '' , $fields = '*' , $extraselect = '' , array $extraparams = null ) {
2010-08-25 08:56:07 +00:00
global $DB , $CFG ;
2007-08-02 23:39:28 +00:00
2006-09-15 14:32:35 +00:00
if ( $get && ! $recordsperpage ) {
debugging ( 'Call to get_users with $get = true no $recordsperpage limit. ' .
'On large installations, this will probably cause an out of memory error. ' .
'Please think again and change your code so that it does not try to ' .
2006-10-01 06:39:20 +00:00
'load so much data into memory.' , DEBUG_DEVELOPER );
2006-09-15 14:32:35 +00:00
}
2003-09-14 04:04:15 +00:00
2008-06-09 19:48:24 +00:00
$fullname = $DB -> sql_fullname ();
2003-03-21 10:10:21 +00:00
2010-08-25 08:56:07 +00:00
$select = " id <> :guestid AND deleted = 0 " ;
$params = array ( 'guestid' => $CFG -> siteguest );
2004-03-20 06:58:52 +00:00
2004-12-22 23:11:27 +00:00
if ( ! empty ( $search )){
$search = trim ( $search );
2010-09-04 12:45:47 +00:00
$select .= " AND ( " . $DB -> sql_like ( $fullname , ':search1' , false ) . " OR " . $DB -> sql_like ( 'email' , ':search2' , false ) . " OR username = :search3) " ;
2008-05-25 09:39:02 +00:00
$params [ 'search1' ] = " % $search % " ;
$params [ 'search2' ] = " % $search % " ;
$params [ 'search3' ] = " $search " ;
2003-03-21 10:10:21 +00:00
}
2003-05-14 15:19:04 +00:00
if ( $confirmed ) {
2008-05-25 09:39:02 +00:00
$select .= " AND confirmed = 1 " ;
2003-05-14 15:19:04 +00:00
}
if ( $exceptions ) {
2011-04-14 15:13:28 +02:00
list ( $exceptions , $eparams ) = $DB -> get_in_or_equal ( $exceptions , SQL_PARAMS_NAMED , 'ex' , false );
2008-05-25 09:39:02 +00:00
$params = $params + $eparams ;
2012-03-05 13:48:55 +00:00
$select .= " AND id $exceptions " ;
2003-05-14 15:19:04 +00:00
}
2004-03-20 06:58:52 +00:00
if ( $firstinitial ) {
2010-09-04 12:45:47 +00:00
$select .= " AND " . $DB -> sql_like ( 'firstname' , ':fni' , false , false );
2008-05-25 09:39:02 +00:00
$params [ 'fni' ] = " $firstinitial % " ;
2004-09-21 11:41:58 +00:00
}
2004-03-20 06:58:52 +00:00
if ( $lastinitial ) {
2010-09-04 12:45:47 +00:00
$select .= " AND " . $DB -> sql_like ( 'lastname' , ':lni' , false , false );
2008-05-25 09:39:02 +00:00
$params [ 'lni' ] = " $lastinitial % " ;
2004-09-21 11:41:58 +00:00
}
2004-03-20 06:58:52 +00:00
2007-11-13 08:43:20 +00:00
if ( $extraselect ) {
2008-05-25 09:39:02 +00:00
$select .= " AND $extraselect " ;
$params = $params + ( array ) $extraparams ;
2007-11-13 08:43:20 +00:00
}
2003-05-14 15:19:04 +00:00
if ( $get ) {
2008-05-25 09:39:02 +00:00
return $DB -> get_records_select ( 'user' , $select , $params , $sort , $fields , $page , $recordsperpage );
2003-05-14 15:19:04 +00:00
} else {
2008-05-25 09:39:02 +00:00
return $DB -> count_records_select ( 'user' , $select , $params );
2003-05-14 15:19:04 +00:00
}
2002-12-20 14:44:14 +00:00
}
2003-05-14 15:19:04 +00:00
2003-09-14 04:04:15 +00:00
/**
2013-03-12 10:00:55 +08:00
* Return filtered ( if provided ) list of users in site , except guest and deleted users .
2009-05-22 08:41:00 +00:00
*
* @ param string $sort An SQL field to sort by
* @ param string $dir The sort direction ASC | DESC
* @ param int $page The page or records to return
* @ param int $recordsperpage The number of records to return per page
* @ param string $search A simple string to search for
* @ param string $firstinitial Users whose first name starts with $firstinitial
* @ param string $lastinitial Users whose last name starts with $lastinitial
* @ param string $extraselect An additional SQL select statement to append to the query
* @ param array $extraparams Additional parameters to use for the above $extraselect
2013-03-12 10:00:55 +08:00
* @ param stdClass $extracontext If specified , will include user 'extra fields'
2011-04-13 15:44:42 +01:00
* as appropriate for current user and given context
2009-05-22 08:41:00 +00:00
* @ return array Array of { @ link $USER } records
2004-09-24 21:28:22 +00:00
*/
2006-09-15 14:32:35 +00:00
function get_users_listing ( $sort = 'lastaccess' , $dir = 'ASC' , $page = 0 , $recordsperpage = 0 ,
2011-04-13 15:44:42 +01:00
$search = '' , $firstinitial = '' , $lastinitial = '' , $extraselect = '' ,
array $extraparams = null , $extracontext = null ) {
2013-03-08 11:43:48 +08:00
global $DB , $CFG ;
2002-12-23 13:48:31 +00:00
2008-06-09 19:48:24 +00:00
$fullname = $DB -> sql_fullname ();
2003-02-18 03:16:07 +00:00
2013-03-08 11:43:48 +08:00
$select = " deleted <> 1 AND id <> :guestid " ;
$params = array ( 'guestid' => $CFG -> siteguest );
2004-03-20 06:58:52 +00:00
2004-12-22 23:11:27 +00:00
if ( ! empty ( $search )) {
$search = trim ( $search );
2010-09-04 11:46:00 +00:00
$select .= " AND ( " . $DB -> sql_like ( $fullname , ':search1' , false , false ) .
" OR " . $DB -> sql_like ( 'email' , ':search2' , false , false ) .
" OR username = :search3) " ;
2008-05-25 09:39:02 +00:00
$params [ 'search1' ] = " % $search % " ;
$params [ 'search2' ] = " % $search % " ;
$params [ 'search3' ] = " $search " ;
2004-03-20 06:58:52 +00:00
}
if ( $firstinitial ) {
2010-09-04 11:46:00 +00:00
$select .= " AND " . $DB -> sql_like ( 'firstname' , ':fni' , false , false );
2008-05-25 09:39:02 +00:00
$params [ 'fni' ] = " $firstinitial % " ;
2004-03-20 06:58:52 +00:00
}
if ( $lastinitial ) {
2010-09-04 11:46:00 +00:00
$select .= " AND " . $DB -> sql_like ( 'lastname' , ':lni' , false , false );
2008-05-25 09:39:02 +00:00
$params [ 'lni' ] = " $lastinitial % " ;
2003-03-21 09:42:42 +00:00
}
2007-11-13 08:43:20 +00:00
if ( $extraselect ) {
2008-05-25 09:39:02 +00:00
$select .= " AND $extraselect " ;
$params = $params + ( array ) $extraparams ;
2007-11-13 08:43:20 +00:00
}
2007-01-04 02:52:44 +00:00
2004-03-20 06:58:52 +00:00
if ( $sort ) {
2008-05-25 09:39:02 +00:00
$sort = " ORDER BY $sort $dir " ;
2004-03-20 06:58:52 +00:00
}
2011-04-13 15:44:42 +01:00
// If a context is specified, get extra user fields that the current user
// is supposed to see.
$extrafields = '' ;
if ( $extracontext ) {
$extrafields = get_extra_user_fields_sql ( $extracontext , '' , '' ,
array ( 'id' , 'username' , 'email' , 'firstname' , 'lastname' , 'city' , 'country' ,
'lastaccess' , 'confirmed' , 'mnethostid' ));
}
2013-04-24 10:12:42 +08:00
$namefields = get_all_user_name_fields ( true );
$extrafields = " $extrafields , $namefields " ;
2011-04-13 15:44:42 +01:00
// warning: will return UNCONFIRMED USERS
2013-04-24 10:12:42 +08:00
return $DB -> get_records_sql ( " SELECT id, username, email, city, country, lastaccess, confirmed, mnethostid, suspended $extrafields
2008-05-25 09:39:02 +00:00
FROM { user }
WHERE $select
$sort " , $params , $page , $recordsperpage );
2002-12-20 14:44:14 +00:00
}
2004-03-20 06:58:52 +00:00
2003-09-14 04:04:15 +00:00
/**
2005-07-12 02:23:58 +00:00
* Full list of users that have confirmed their accounts .
2004-09-24 21:28:22 +00:00
*
2009-05-22 08:41:00 +00:00
* @ global object
2008-05-25 09:39:02 +00:00
* @ return array of unconfirmed users
2004-09-24 21:28:22 +00:00
*/
2002-12-20 14:44:14 +00:00
function get_users_confirmed () {
2010-08-25 08:56:07 +00:00
global $DB , $CFG ;
2008-05-25 09:39:02 +00:00
return $DB -> get_records_sql ( " SELECT *
FROM { user }
2010-08-25 08:56:07 +00:00
WHERE confirmed = 1 AND deleted = 0 AND id <> ? " , array( $CFG->siteguest ));
2002-12-20 14:44:14 +00:00
}
2003-08-15 13:59:24 +00:00
/// OTHER SITE AND COURSE FUNCTIONS /////////////////////////////////////////////
2003-09-14 04:04:15 +00:00
/**
2004-09-24 21:28:22 +00:00
* Returns $course object of the top - level site .
*
2009-11-01 09:10:09 +00:00
* @ return object A { @ link $COURSE } object for the site , exception if not found
2004-09-24 21:28:22 +00:00
*/
2005-03-03 04:41:46 +00:00
function get_site () {
2008-05-25 09:39:02 +00:00
global $SITE , $DB ;
2005-03-03 04:41:46 +00:00
if ( ! empty ( $SITE -> id )) { // We already have a global to use, so return that
return $SITE ;
}
2003-08-15 13:59:24 +00:00
2008-05-25 09:39:02 +00:00
if ( $course = $DB -> get_record ( 'course' , array ( 'category' => 0 ))) {
2003-08-15 13:59:24 +00:00
return $course ;
} else {
2009-11-01 09:10:09 +00:00
// course table exists, but the site is not there,
// unfortunately there is no automatic way to recover
throw new moodle_exception ( 'nosite' , 'error' );
2003-08-15 13:59:24 +00:00
}
}
2013-05-17 16:43:51 +01:00
/**
* Gets a course object from database . If the course id corresponds to an
* already - loaded $COURSE or $SITE object , then the loaded object will be used ,
* saving a database query .
*
* If it reuses an existing object , by default the object will be cloned . This
* means you can modify the object safely without affecting other code .
*
* @ param int $courseid Course id
* @ param bool $clone If true ( default ), makes a clone of the record
* @ return stdClass A course object
* @ throws dml_exception If not found in database
*/
function get_course ( $courseid , $clone = true ) {
global $DB , $COURSE , $SITE ;
if ( ! empty ( $COURSE -> id ) && $COURSE -> id == $courseid ) {
return $clone ? clone ( $COURSE ) : $COURSE ;
} else if ( ! empty ( $SITE -> id ) && $SITE -> id == $courseid ) {
return $clone ? clone ( $SITE ) : $SITE ;
} else {
return $DB -> get_record ( 'course' , array ( 'id' => $courseid ), '*' , MUST_EXIST );
}
}
2003-09-14 04:04:15 +00:00
/**
2006-09-10 07:07:52 +00:00
* Returns list of courses , for whole site , or category
*
* Returns list of courses , for whole site , or category
2007-09-20 13:15:26 +00:00
* Important : Using c .* for fields is extremely expensive because
2006-09-10 07:07:52 +00:00
* we are using distinct . You almost _NEVER_ need all the fields
* in such a large SELECT
*
2018-06-21 13:34:59 +08:00
* Consider using core_course_category :: get_courses ()
* or core_course_category :: search_courses () instead since they use caching .
*
2009-05-22 08:41:00 +00:00
* @ global object
* @ global object
* @ global object
* @ uses CONTEXT_COURSE
* @ param string | int $categoryid Either a category id or 'all' for everything
* @ param string $sort A field and direction to sort by
* @ param string $fields The additional fields to return
* @ return array Array of courses
2006-09-10 07:07:52 +00:00
*/
2004-11-18 02:31:53 +00:00
function get_courses ( $categoryid = " all " , $sort = " c.sortorder ASC " , $fields = " c.* " ) {
2003-08-15 13:59:24 +00:00
2008-05-25 15:16:17 +00:00
global $USER , $CFG , $DB ;
2007-08-02 23:39:28 +00:00
2008-05-25 15:16:17 +00:00
$params = array ();
if ( $categoryid !== " all " && is_numeric ( $categoryid )) {
$categoryselect = " WHERE c.category = :catid " ;
$params [ 'catid' ] = $categoryid ;
2006-09-15 09:08:48 +00:00
} else {
2007-08-02 23:39:28 +00:00
$categoryselect = " " ;
2006-09-17 18:07:35 +00:00
}
if ( empty ( $sort )) {
$sortstatement = " " ;
} else {
$sortstatement = " ORDER BY $sort " ;
}
$visiblecourses = array ();
2007-08-02 23:39:28 +00:00
2013-07-05 13:02:00 +08:00
$ccselect = ', ' . context_helper :: get_preload_record_columns_sql ( 'ctx' );
$ccjoin = " LEFT JOIN { context} ctx ON (ctx.instanceid = c.id AND ctx.contextlevel = :contextlevel) " ;
$params [ 'contextlevel' ] = CONTEXT_COURSE ;
2010-03-31 07:41:31 +00:00
$sql = " SELECT $fields $ccselect
2008-05-25 15:16:17 +00:00
FROM { course } c
2010-03-31 07:41:31 +00:00
$ccjoin
2008-05-25 15:16:17 +00:00
$categoryselect
$sortstatement " ;
2006-09-15 09:08:48 +00:00
// pull out all course matching the cat
2008-05-25 15:16:17 +00:00
if ( $courses = $DB -> get_records_sql ( $sql , $params )) {
2006-09-17 18:07:35 +00:00
// loop throught them
foreach ( $courses as $course ) {
2013-07-05 11:48:36 +08:00
context_helper :: preload_from_record ( $course );
2018-06-21 13:34:59 +08:00
if ( core_course_category :: can_view_course_info ( $course )) {
2008-05-25 15:16:17 +00:00
$visiblecourses [ $course -> id ] = $course ;
2007-08-02 23:39:28 +00:00
}
2006-09-17 18:07:35 +00:00
}
2004-11-18 02:31:53 +00:00
}
2006-09-15 09:08:48 +00:00
return $visiblecourses ;
2003-08-15 13:59:24 +00:00
}
2003-09-14 04:04:15 +00:00
/**
2005-07-12 02:23:58 +00:00
* A list of courses that match a search
2004-09-24 21:28:22 +00:00
*
2009-05-22 08:41:00 +00:00
* @ global object
* @ global object
* @ param array $searchterms An array of search criteria
* @ param string $sort A field and direction to sort by
* @ param int $page The page number to get
* @ param int $recordsperpage The number of records per page
* @ param int $totalcount Passed in by reference .
2015-09-04 14:59:04 +08:00
* @ param array $requiredcapabilities Extra list of capabilities used to filter courses
2019-04-07 23:36:34 +02:00
* @ param array $searchcond additional search conditions , for example [ 'c.enablecompletion = :p1' ]
* @ param array $params named parameters for additional search conditions , for example [ 'p1' => 1 ]
2018-06-21 13:34:59 +08:00
* @ return stdClass [] { @ link $COURSE } records
2004-09-24 21:28:22 +00:00
*/
2015-09-04 14:59:04 +08:00
function get_courses_search ( $searchterms , $sort , $page , $recordsperpage , & $totalcount ,
2019-04-07 23:36:34 +02:00
$requiredcapabilities = array (), $searchcond = [], $params = []) {
2008-05-25 15:16:17 +00:00
global $CFG , $DB ;
2003-08-15 13:59:24 +00:00
2008-05-25 20:43:46 +00:00
if ( $DB -> sql_regex_supported ()) {
$REGEXP = $DB -> sql_regex ( true );
$NOTREGEXP = $DB -> sql_regex ( false );
2003-08-15 13:59:24 +00:00
}
2008-05-25 20:43:46 +00:00
$i = 0 ;
2003-08-15 13:59:24 +00:00
2011-10-26 19:04:31 +02:00
// Thanks Oracle for your non-ansi concat and type limits in coalesce. MDL-29912
if ( $DB -> get_dbfamily () == 'oracle' ) {
2013-04-23 12:36:50 +02:00
$concat = " (c.summary|| ' ' || c.fullname || ' ' || c.idnumber || ' ' || c.shortname) " ;
2011-10-26 19:04:31 +02:00
} else {
2013-04-23 12:36:50 +02:00
$concat = $DB -> sql_concat ( " COALESCE(c.summary, '') " , " ' ' " , 'c.fullname' , " ' ' " , 'c.idnumber' , " ' ' " , 'c.shortname' );
2011-10-26 19:04:31 +02:00
}
2008-05-25 15:16:17 +00:00
2003-08-15 13:59:24 +00:00
foreach ( $searchterms as $searchterm ) {
2008-05-25 20:43:46 +00:00
$i ++ ;
2006-10-31 19:54:46 +00:00
2010-09-04 12:45:47 +00:00
$NOT = false ; /// Initially we aren't going to perform NOT LIKE searches, only MSSQL and Oracle
2008-01-01 12:03:00 +00:00
/// will use it to simulate the "-" operator with LIKE clause
2006-10-31 19:54:46 +00:00
/// Under Oracle and MSSQL, trim the + and - operators and perform
2008-01-01 12:03:00 +00:00
/// simpler LIKE (or NOT LIKE) queries
2008-05-25 20:43:46 +00:00
if ( ! $DB -> sql_regex_supported ()) {
2008-01-01 12:03:00 +00:00
if ( substr ( $searchterm , 0 , 1 ) == '-' ) {
2010-09-04 12:45:47 +00:00
$NOT = true ;
2008-01-01 12:03:00 +00:00
}
2006-10-31 19:54:46 +00:00
$searchterm = trim ( $searchterm , '+-' );
}
2008-05-25 20:43:46 +00:00
// TODO: +- may not work for non latin languages
2008-05-25 15:16:17 +00:00
2004-09-22 16:15:23 +00:00
if ( substr ( $searchterm , 0 , 1 ) == '+' ) {
2008-05-25 20:43:46 +00:00
$searchterm = trim ( $searchterm , '+-' );
$searchterm = preg_quote ( $searchterm , '|' );
$searchcond [] = " $concat $REGEXP :ss $i " ;
$params [ 'ss' . $i ] = " (^|[^a-zA-Z0-9]) $searchterm ([^a-zA-Z0-9]| $ ) " ;
2017-03-14 19:07:09 +01:00
} else if (( substr ( $searchterm , 0 , 1 ) == " - " ) && ( core_text :: strlen ( $searchterm ) > 1 )) {
2008-05-25 20:43:46 +00:00
$searchterm = trim ( $searchterm , '+-' );
$searchterm = preg_quote ( $searchterm , '|' );
$searchcond [] = " $concat $NOTREGEXP :ss $i " ;
$params [ 'ss' . $i ] = " (^|[^a-zA-Z0-9]) $searchterm ([^a-zA-Z0-9]| $ ) " ;
2003-08-19 05:32:20 +00:00
} else {
2010-09-04 14:52:47 +00:00
$searchcond [] = $DB -> sql_like ( $concat , " :ss $i " , false , true , $NOT );
2008-05-25 20:43:46 +00:00
$params [ 'ss' . $i ] = " % $searchterm % " ;
2003-08-19 05:32:20 +00:00
}
2003-08-15 13:59:24 +00:00
}
2008-05-25 20:43:46 +00:00
if ( empty ( $searchcond )) {
2015-09-04 14:59:04 +08:00
$searchcond = array ( '1 = 1' );
2008-05-25 20:43:46 +00:00
}
$searchcond = implode ( " AND " , $searchcond );
2010-12-30 01:48:43 +01:00
$courses = array ();
$c = 0 ; // counts how many visible courses we've seen
// Tiki pagination
$limitfrom = $page * $recordsperpage ;
$limitto = $limitfrom + $recordsperpage ;
2013-07-05 13:02:00 +08:00
$ccselect = ', ' . context_helper :: get_preload_record_columns_sql ( 'ctx' );
$ccjoin = " LEFT JOIN { context} ctx ON (ctx.instanceid = c.id AND ctx.contextlevel = :contextlevel) " ;
$params [ 'contextlevel' ] = CONTEXT_COURSE ;
2013-07-25 21:53:50 +10:00
$sql = " SELECT c.* $ccselect
2008-05-25 15:16:17 +00:00
FROM { course } c
2010-03-31 07:41:31 +00:00
$ccjoin
2008-05-25 20:43:46 +00:00
WHERE $searchcond AND c . id <> " .SITEID. "
ORDER BY $sort " ;
2007-09-19 07:08:37 +00:00
2018-06-21 13:34:59 +08:00
$mycourses = enrol_get_my_courses ();
2010-12-30 01:48:43 +01:00
$rs = $DB -> get_recordset_sql ( $sql , $params );
foreach ( $rs as $course ) {
2015-09-04 14:59:04 +08:00
// Preload contexts only for hidden courses or courses we need to return.
context_helper :: preload_from_record ( $course );
$coursecontext = context_course :: instance ( $course -> id );
2018-06-21 13:34:59 +08:00
if ( ! array_key_exists ( $course -> id , $mycourses ) && ! core_course_category :: can_view_course_info ( $course )) {
2015-09-04 14:59:04 +08:00
continue ;
}
if ( ! empty ( $requiredcapabilities )) {
if ( ! has_all_capabilities ( $requiredcapabilities , $coursecontext )) {
2013-03-05 11:41:16 +11:00
continue ;
2003-08-15 13:59:24 +00:00
}
}
2013-03-05 11:41:16 +11:00
// Don't exit this loop till the end
// we need to count all the visible courses
// to update $totalcount
if ( $c >= $limitfrom && $c < $limitto ) {
$courses [ $course -> id ] = $course ;
}
$c ++ ;
2003-08-15 13:59:24 +00:00
}
2010-12-30 01:48:43 +01:00
$rs -> close ();
2003-08-15 13:59:24 +00:00
2007-09-19 07:08:37 +00:00
// our caller expects 2 bits of data - our return
// array, and an updated $totalcount
$totalcount = $c ;
2003-08-15 13:59:24 +00:00
return $courses ;
}
2008-06-16 14:25:53 +00:00
/**
* Fixes course category and course sortorder , also verifies category and course parents and paths .
2008-08-27 11:31:37 +00:00
* ( circular references are not fixed )
2009-05-22 08:41:00 +00:00
*
* @ global object
* @ global object
* @ uses MAX_COURSES_IN_CATEGORY
* @ uses MAX_COURSE_CATEGORIES
* @ uses SITEID
* @ uses CONTEXT_COURSE
* @ return void
2008-06-16 14:25:53 +00:00
*/
function fix_course_sortorder () {
global $DB , $SITE ;
//WARNING: this is PHP5 only code!
2013-02-28 12:29:15 +11:00
// if there are any changes made to courses or categories we will trigger
// the cache events to purge all cached courses/categories data
$cacheevents = array ();
2008-06-16 14:25:53 +00:00
if ( $unsorted = $DB -> get_records ( 'course_categories' , array ( 'sortorder' => 0 ))) {
//move all categories that are not sorted yet to the end
$DB -> set_field ( 'course_categories' , 'sortorder' , MAX_COURSES_IN_CATEGORY * MAX_COURSE_CATEGORIES , array ( 'sortorder' => 0 ));
2013-02-28 12:29:15 +11:00
$cacheevents [ 'changesincoursecat' ] = true ;
2008-06-16 14:25:53 +00:00
}
$allcats = $DB -> get_records ( 'course_categories' , null , 'sortorder, id' , 'id, sortorder, parent, depth, path' );
$topcats = array ();
$brokencats = array ();
foreach ( $allcats as $cat ) {
$sortorder = ( int ) $cat -> sortorder ;
if ( ! $cat -> parent ) {
while ( isset ( $topcats [ $sortorder ])) {
$sortorder ++ ;
}
$topcats [ $sortorder ] = $cat ;
continue ;
}
if ( ! isset ( $allcats [ $cat -> parent ])) {
$brokencats [] = $cat ;
continue ;
2007-09-19 07:27:08 +00:00
}
2008-06-16 14:25:53 +00:00
if ( ! isset ( $allcats [ $cat -> parent ] -> children )) {
$allcats [ $cat -> parent ] -> children = array ();
2007-09-19 07:27:08 +00:00
}
2008-06-16 14:25:53 +00:00
while ( isset ( $allcats [ $cat -> parent ] -> children [ $sortorder ])) {
$sortorder ++ ;
}
$allcats [ $cat -> parent ] -> children [ $sortorder ] = $cat ;
2005-08-16 23:15:58 +00:00
}
2008-06-16 14:25:53 +00:00
unset ( $allcats );
2005-01-13 02:34:45 +00:00
2008-06-16 14:25:53 +00:00
// add broken cats to category tree
if ( $brokencats ) {
$defaultcat = reset ( $topcats );
foreach ( $brokencats as $cat ) {
$topcats [] = $cat ;
2008-08-27 11:31:37 +00:00
}
2004-11-17 06:57:28 +00:00
}
2008-06-16 14:25:53 +00:00
// now walk recursively the tree and fix any problems found
$sortorder = 0 ;
$fixcontexts = array ();
2013-02-28 12:29:15 +11:00
if ( _fix_course_cats ( $topcats , $sortorder , 0 , 0 , '' , $fixcontexts )) {
$cacheevents [ 'changesincoursecat' ] = true ;
}
2008-06-16 14:25:53 +00:00
// detect if there are "multiple" frontpage courses and fix them if needed
$frontcourses = $DB -> get_records ( 'course' , array ( 'category' => 0 ), 'id' );
if ( count ( $frontcourses ) > 1 ) {
if ( isset ( $frontcourses [ SITEID ])) {
$frontcourse = $frontcourses [ SITEID ];
unset ( $frontcourses [ SITEID ]);
} else {
$frontcourse = array_shift ( $frontcourses );
}
$defaultcat = reset ( $topcats );
foreach ( $frontcourses as $course ) {
$DB -> set_field ( 'course' , 'category' , $defaultcat -> id , array ( 'id' => $course -> id ));
2012-07-25 16:25:55 +08:00
$context = context_course :: instance ( $course -> id );
2008-06-16 14:25:53 +00:00
$fixcontexts [ $context -> id ] = $context ;
2013-02-28 12:29:15 +11:00
$cacheevents [ 'changesincourse' ] = true ;
2008-06-16 14:25:53 +00:00
}
unset ( $frontcourses );
} else {
$frontcourse = reset ( $frontcourses );
2005-02-23 01:49:22 +00:00
}
2008-06-16 14:25:53 +00:00
// now fix the paths and depths in context table if needed
if ( $fixcontexts ) {
2011-10-14 12:48:00 +02:00
foreach ( $fixcontexts as $fixcontext ) {
$fixcontext -> reset_paths ( false );
}
context_helper :: build_all_paths ( false );
unset ( $fixcontexts );
2013-02-28 12:29:15 +11:00
$cacheevents [ 'changesincourse' ] = true ;
$cacheevents [ 'changesincoursecat' ] = true ;
2005-01-13 02:34:45 +00:00
}
2007-08-02 23:39:28 +00:00
2008-06-16 14:25:53 +00:00
// release memory
unset ( $topcats );
unset ( $brokencats );
unset ( $fixcontexts );
// fix frontpage course sortorder
if ( $frontcourse -> sortorder != 1 ) {
$DB -> set_field ( 'course' , 'sortorder' , 1 , array ( 'id' => $frontcourse -> id ));
2013-02-28 12:29:15 +11:00
$cacheevents [ 'changesincourse' ] = true ;
2005-01-13 02:34:45 +00:00
}
2008-06-16 14:25:53 +00:00
// now fix the course counts in category records if needed
$sql = " SELECT cc.id, cc.coursecount, COUNT(c.id) AS newcount
FROM { course_categories } cc
LEFT JOIN { course } c ON c . category = cc . id
GROUP BY cc . id , cc . coursecount
HAVING cc . coursecount <> COUNT ( c . id ) " ;
2004-11-17 06:57:28 +00:00
2008-06-16 14:25:53 +00:00
if ( $updatecounts = $DB -> get_records_sql ( $sql )) {
2011-01-07 11:54:41 +08:00
// categories with more courses than MAX_COURSES_IN_CATEGORY
$categories = array ();
2008-06-16 14:25:53 +00:00
foreach ( $updatecounts as $cat ) {
$cat -> coursecount = $cat -> newcount ;
2011-01-07 11:54:41 +08:00
if ( $cat -> coursecount >= MAX_COURSES_IN_CATEGORY ) {
$categories [] = $cat -> id ;
}
2008-06-16 14:25:53 +00:00
unset ( $cat -> newcount );
$DB -> update_record_raw ( 'course_categories' , $cat , true );
2008-08-27 11:31:37 +00:00
}
2011-01-07 11:54:41 +08:00
if ( ! empty ( $categories )) {
$str = implode ( ', ' , $categories );
debugging ( " The number of courses (category id: $str ) has reached MAX_COURSES_IN_CATEGORY ( " . MAX_COURSES_IN_CATEGORY . " ), it will cause a sorting performance issue, please increase the value of MAX_COURSES_IN_CATEGORY in lib/datalib.php file. See tracker issue: MDL-25669 " , DEBUG_DEVELOPER );
}
2013-02-28 12:29:15 +11:00
$cacheevents [ 'changesincoursecat' ] = true ;
2003-08-15 13:59:24 +00:00
}
2004-09-21 11:41:58 +00:00
2008-06-16 14:25:53 +00:00
// now make sure that sortorders in course table are withing the category sortorder ranges
course categories: Fix many bugs with category editing and permissions. Clean up code.
Bugs: MDL-17479, MDL-16426, MDL-16063, MDL-16013, MDL-15658, MDL-15556, MDL-15161, MDL-14925, MDL-13742, MDL-11557.
* Simplify category editing permissions to just moodle/category:manage and moodle/category:seehiddencategories.
* Enforce those correctly. (Note MDL 17502 is still outstanding.)
* Don't screw up category sort order when you just edit name or description.
* Niceties like where redirects go when you cancel or submit forms.
* Make sure a global course creator can see the site admin block.
* Don't allow a category to be made the child of one of its children!
* General code cleanup to bring key files more in line with best pracitice.
Apologies for the fact it is one big patch, rather than a series of smaller patches. However, categoryedit.php, category.php and index.php where in pretty bad shape and needed significant cleaning up. categoryedit.php, in particular, was almost completely rewritten.
Merged from MOODLE_19_STABLE.
2008-12-04 08:53:10 +00:00
$sql = " SELECT DISTINCT cc.id, cc.sortorder
2008-06-16 14:25:53 +00:00
FROM { course_categories } cc
JOIN { course } c ON c . category = cc . id
WHERE c . sortorder < cc . sortorder OR c . sortorder > cc . sortorder + " .MAX_COURSES_IN_CATEGORY;
if ( $fixcategories = $DB -> get_records_sql ( $sql )) {
//fix the course sortorder ranges
foreach ( $fixcategories as $cat ) {
$sql = " UPDATE { course}
2008-11-26 08:51:08 +00:00
SET sortorder = " . $DB->sql_modulo ('sortorder', MAX_COURSES_IN_CATEGORY). " + ?
2008-06-16 14:25:53 +00:00
WHERE category = ? " ;
$DB -> execute ( $sql , array ( $cat -> sortorder , $cat -> id ));
}
2013-02-28 12:29:15 +11:00
$cacheevents [ 'changesincoursecat' ] = true ;
2005-02-23 01:49:22 +00:00
}
2008-06-16 14:25:53 +00:00
unset ( $fixcategories );
// categories having courses with sortorder duplicates or having gaps in sortorder
$sql = " SELECT DISTINCT c1.category AS id , cc.sortorder
FROM { course } c1
JOIN { course } c2 ON c1 . sortorder = c2 . sortorder
JOIN { course_categories } cc ON ( c1 . category = cc . id )
WHERE c1 . id <> c2 . id " ;
$fixcategories = $DB -> get_records_sql ( $sql );
$sql = " SELECT cc.id, cc.sortorder, cc.coursecount, MAX(c.sortorder) AS maxsort, MIN(c.sortorder) AS minsort
FROM { course_categories } cc
JOIN { course } c ON c . category = cc . id
GROUP BY cc . id , cc . sortorder , cc . coursecount
HAVING ( MAX ( c . sortorder ) <> cc . sortorder + cc . coursecount ) OR ( MIN ( c . sortorder ) <> cc . sortorder + 1 ) " ;
$gapcategories = $DB -> get_records_sql ( $sql );
foreach ( $gapcategories as $cat ) {
if ( isset ( $fixcategories [ $cat -> id ])) {
// duplicates detected already
} else if ( $cat -> minsort == $cat -> sortorder and $cat -> maxsort == $cat -> sortorder + $cat -> coursecount - 1 ) {
// easy - new course inserted with sortorder 0, the rest is ok
$sql = " UPDATE { course}
SET sortorder = sortorder + 1
WHERE category = ? " ;
$DB -> execute ( $sql , array ( $cat -> id ));
2005-01-25 05:27:41 +00:00
2008-06-16 14:25:53 +00:00
} else {
// it needs full resorting
$fixcategories [ $cat -> id ] = $cat ;
2004-05-30 00:33:45 +00:00
}
2013-02-28 12:29:15 +11:00
$cacheevents [ 'changesincourse' ] = true ;
2004-05-30 00:33:45 +00:00
}
2008-06-16 14:25:53 +00:00
unset ( $gapcategories );
2004-09-21 11:41:58 +00:00
2008-06-16 14:25:53 +00:00
// fix course sortorders in problematic categories only
foreach ( $fixcategories as $cat ) {
$i = 1 ;
$courses = $DB -> get_records ( 'course' , array ( 'category' => $cat -> id ), 'sortorder ASC, id DESC' , 'id, sortorder' );
foreach ( $courses as $course ) {
if ( $course -> sortorder != $cat -> sortorder + $i ) {
$course -> sortorder = $cat -> sortorder + $i ;
2008-08-27 11:31:37 +00:00
$DB -> update_record_raw ( 'course' , $course , true );
2013-02-28 12:29:15 +11:00
$cacheevents [ 'changesincourse' ] = true ;
2008-06-16 14:25:53 +00:00
}
$i ++ ;
}
}
2013-02-21 09:57:40 +11:00
2013-02-28 12:29:15 +11:00
// advise all caches that need to be rebuilt
foreach ( array_keys ( $cacheevents ) as $event ) {
cache_helper :: purge_by_event ( $event );
}
2003-08-15 13:59:24 +00:00
}
2007-09-12 02:56:36 +00:00
/**
2008-06-16 14:25:53 +00:00
* Internal recursive category verification function , do not use directly !
2009-05-22 08:41:00 +00:00
*
* @ todo Document the arguments of this function better
*
* @ global object
* @ uses MAX_COURSES_IN_CATEGORY
* @ uses CONTEXT_COURSECAT
* @ param array $children
* @ param int $sortorder
* @ param string $parent
* @ param int $depth
* @ param string $path
* @ param array $fixcontexts
2013-02-28 12:29:15 +11:00
* @ return bool if changes were made
2008-06-16 14:25:53 +00:00
*/
function _fix_course_cats ( $children , & $sortorder , $parent , $depth , $path , & $fixcontexts ) {
2008-05-30 16:47:21 +00:00
global $DB ;
2007-09-12 02:56:36 +00:00
2008-06-16 14:25:53 +00:00
$depth ++ ;
2013-02-28 12:29:15 +11:00
$changesmade = false ;
2008-05-30 16:47:21 +00:00
2008-06-16 14:25:53 +00:00
foreach ( $children as $cat ) {
$sortorder = $sortorder + MAX_COURSES_IN_CATEGORY ;
$update = false ;
if ( $parent != $cat -> parent or $depth != $cat -> depth or $path . '/' . $cat -> id != $cat -> path ) {
$cat -> parent = $parent ;
$cat -> depth = $depth ;
$cat -> path = $path . '/' . $cat -> id ;
$update = true ;
2008-05-30 16:47:21 +00:00
2008-06-16 14:25:53 +00:00
// make sure context caches are rebuild and dirty contexts marked
2012-07-25 16:25:55 +08:00
$context = context_coursecat :: instance ( $cat -> id );
2008-06-16 14:25:53 +00:00
$fixcontexts [ $context -> id ] = $context ;
}
if ( $cat -> sortorder != $sortorder ) {
$cat -> sortorder = $sortorder ;
$update = true ;
}
if ( $update ) {
$DB -> update_record ( 'course_categories' , $cat , true );
2013-02-28 12:29:15 +11:00
$changesmade = true ;
2008-06-16 14:25:53 +00:00
}
if ( isset ( $cat -> children )) {
2013-02-28 12:29:15 +11:00
if ( _fix_course_cats ( $cat -> children , $sortorder , $cat -> id , $cat -> depth , $cat -> path , $fixcontexts )) {
$changesmade = true ;
}
2007-09-12 02:56:36 +00:00
}
}
2013-02-28 12:29:15 +11:00
return $changesmade ;
2007-09-12 02:56:36 +00:00
}
2007-01-19 08:57:13 +00:00
/**
* List of remote courses that a user has access to via MNET .
* Works only on the IDP
*
2009-05-22 08:41:00 +00:00
* @ global object
* @ global object
* @ param int @ userid The user id to get remote courses for
* @ return array Array of { @ link $COURSE } of course objects
2007-01-19 08:57:13 +00:00
*/
function get_my_remotecourses ( $userid = 0 ) {
2008-05-30 16:47:21 +00:00
global $DB , $USER ;
2007-01-19 08:57:13 +00:00
if ( empty ( $userid )) {
$userid = $USER -> id ;
}
2010-07-20 09:02:23 +00:00
// we can not use SELECT DISTINCT + text field (summary) because of MS SQL and Oracle, subselect used therefore
$sql = " SELECT c.id, c.remoteid, c.shortname, c.fullname,
2010-07-17 22:34:12 +00:00
c . hostid , c . summary , c . summaryformat , c . categoryname AS cat_name ,
2007-01-19 09:23:47 +00:00
h . name AS hostname
2010-07-17 22:34:12 +00:00
FROM { mnetservice_enrol_courses } c
2010-07-20 09:02:23 +00:00
JOIN ( SELECT DISTINCT hostid , remotecourseid
FROM { mnetservice_enrol_enrolments }
WHERE userid = ?
) e ON ( e . hostid = c . hostid AND e . remotecourseid = c . remoteid )
JOIN { mnet_host } h ON h . id = c . hostid " ;
2007-01-19 08:57:13 +00:00
2008-05-30 16:47:21 +00:00
return $DB -> get_records_sql ( $sql , array ( $userid ));
2007-01-19 08:57:13 +00:00
}
/**
* List of remote hosts that a user has access to via MNET .
* Works on the SP
*
2009-05-22 08:41:00 +00:00
* @ global object
* @ global object
* @ return array | bool Array of host objects or false
2007-01-19 08:57:13 +00:00
*/
function get_my_remotehosts () {
global $CFG , $USER ;
if ( $USER -> mnethostid == $CFG -> mnet_localhost_id ) {
return false ; // Return nothing on the IDP
}
if ( ! empty ( $USER -> mnet_foreign_host_array ) && is_array ( $USER -> mnet_foreign_host_array )) {
return $USER -> mnet_foreign_host_array ;
}
return false ;
}
2004-09-24 21:28:22 +00:00
2003-09-14 04:04:15 +00:00
/**
2004-09-24 21:28:22 +00:00
* Returns a menu of all available scales from the site as well as the given course
*
2009-05-22 08:41:00 +00:00
* @ global object
2004-09-24 21:28:22 +00:00
* @ param int $courseid The id of the course as found in the 'course' table .
2009-05-22 08:41:00 +00:00
* @ return array
2004-09-24 21:28:22 +00:00
*/
2003-08-15 13:59:24 +00:00
function get_scales_menu ( $courseid = 0 ) {
2008-05-30 16:47:21 +00:00
global $DB ;
2003-08-15 13:59:24 +00:00
2018-08-22 16:20:36 +08:00
$sql = " SELECT id, name, courseid
2008-05-30 16:47:21 +00:00
FROM { scale }
WHERE courseid = 0 or courseid = ?
2003-08-15 13:59:24 +00:00
ORDER BY courseid ASC , name ASC " ;
2008-05-30 16:47:21 +00:00
$params = array ( $courseid );
2018-06-14 16:10:35 +02:00
$scales = array ();
2018-08-22 16:20:36 +08:00
$results = $DB -> get_records_sql ( $sql , $params );
foreach ( $results as $index => $record ) {
$context = empty ( $record -> courseid ) ? context_system :: instance () : context_course :: instance ( $record -> courseid );
$scales [ $index ] = format_string ( $record -> name , false , [ " context " => $context ]);
2018-06-14 16:10:35 +02:00
}
2018-08-22 16:20:36 +08:00
// Format: [id => 'scale name'].
2018-06-14 16:10:35 +02:00
return $scales ;
2003-08-15 13:59:24 +00:00
}
2013-09-03 16:42:42 +02:00
/**
* Increment standard revision field .
*
* The revision are based on current time and are incrementing .
* There is a protection for runaway revisions , it may not go further than
* one hour into future .
*
* The field has to be XMLDB_TYPE_INTEGER with size 10.
*
* @ param string $table
* @ param string $field name of the field containing revision
* @ param string $select use empty string when updating all records
* @ param array $params optional select parameters
*/
function increment_revision_number ( $table , $field , $select , array $params = null ) {
global $DB ;
$now = time ();
$sql = " UPDATE { { $table } }
SET $field = ( CASE
WHEN $field IS NULL THEN $now
WHEN $field < $now THEN $now
WHEN $field > $now + 3600 THEN $now
ELSE $field + 1 END ) " ;
if ( $select ) {
$sql = $sql . " WHERE $select " ;
}
$DB -> execute ( $sql , $params );
}
2005-04-10 09:15:15 +00:00
2002-12-17 04:41:18 +00:00
/// MODULE FUNCTIONS /////////////////////////////////////////////////
2003-09-14 04:04:15 +00:00
/**
2004-09-24 21:28:22 +00:00
* Just gets a raw list of all modules in a course
*
2009-05-22 08:41:00 +00:00
* @ global object
2004-09-24 21:28:22 +00:00
* @ param int $courseid The id of the course as found in the 'course' table .
2009-05-22 08:41:00 +00:00
* @ return array
2004-09-24 21:28:22 +00:00
*/
2002-12-20 14:44:14 +00:00
function get_course_mods ( $courseid ) {
2008-05-30 16:47:21 +00:00
global $DB ;
2002-12-20 14:44:14 +00:00
2006-02-14 02:01:31 +00:00
if ( empty ( $courseid )) {
return false ; // avoid warnings
}
2008-05-30 16:47:21 +00:00
return $DB -> get_records_sql ( " SELECT cm.*, m.name as modname
FROM { modules } m , { course_modules } cm
WHERE cm . course = ? AND cm . module = m . id AND m . visible = 1 " ,
array ( $courseid )); // no disabled mods
2002-12-20 14:44:14 +00:00
}
2004-09-24 21:28:22 +00:00
2003-09-14 04:04:15 +00:00
/**
2006-08-08 22:09:55 +00:00
* Given an id of a course module , finds the coursemodule description
2004-09-24 21:28:22 +00:00
*
2014-03-14 15:01:40 +08:00
* Please note that this function performs 1 - 2 DB queries . When possible use cached
* course modinfo . For example get_fast_modinfo ( $courseorid ) -> get_cm ( $cmid )
* See also { @ link cm_info :: get_course_module_record ()}
*
2009-05-22 08:41:00 +00:00
* @ global object
2008-08-27 11:31:37 +00:00
* @ param string $modulename name of module type , eg . resource , assignment , ... ( optional , slower and less safe if not specified )
2006-08-08 22:09:55 +00:00
* @ param int $cmid course module id ( id in course_modules table )
* @ param int $courseid optional course id for extra validation
2008-08-27 11:31:37 +00:00
* @ param bool $sectionnum include relative section number ( 0 , 1 , 2 ... )
2009-07-04 11:15:16 +00:00
* @ param int $strictness IGNORE_MISSING means compatible mode , false returned if record not found , debug message if more found ;
* IGNORE_MULTIPLE means return first , ignore multiple records found ( not recommended );
* MUST_EXIST means throw exception if no record or multiple records found
2010-09-18 13:59:39 +00:00
* @ return stdClass
2006-08-08 22:09:55 +00:00
*/
2009-07-04 11:15:16 +00:00
function get_coursemodule_from_id ( $modulename , $cmid , $courseid = 0 , $sectionnum = false , $strictness = IGNORE_MISSING ) {
2008-05-30 16:47:21 +00:00
global $DB ;
2006-08-08 22:09:55 +00:00
2008-08-27 11:31:37 +00:00
$params = array ( 'cmid' => $cmid );
if ( ! $modulename ) {
if ( ! $modulename = $DB -> get_field_sql ( " SELECT md.name
FROM { modules } md
JOIN { course_modules } cm ON cm . module = md . id
2009-07-04 11:15:16 +00:00
WHERE cm . id = : cmid " , $params , $strictness )) {
2008-08-27 11:31:37 +00:00
return false ;
}
2014-08-07 11:13:22 +12:00
} else {
if ( ! core_component :: is_valid_plugin_name ( 'mod' , $modulename )) {
throw new coding_exception ( 'Invalid modulename parameter' );
}
2008-08-27 11:31:37 +00:00
}
$params [ 'modulename' ] = $modulename ;
2008-05-30 16:51:01 +00:00
$courseselect = " " ;
2008-08-27 11:31:37 +00:00
$sectionfield = " " ;
$sectionjoin = " " ;
2006-08-08 22:09:55 +00:00
2008-05-30 16:47:21 +00:00
if ( $courseid ) {
2008-08-27 11:31:37 +00:00
$courseselect = " AND cm.course = :courseid " ;
2008-05-30 16:47:21 +00:00
$params [ 'courseid' ] = $courseid ;
2008-05-30 16:51:01 +00:00
}
2008-05-30 16:47:21 +00:00
2008-08-27 11:31:37 +00:00
if ( $sectionnum ) {
$sectionfield = " , cw.section AS sectionnum " ;
$sectionjoin = " LEFT JOIN { course_sections} cw ON cw.id = cm.section " ;
}
$sql = " SELECT cm.*, m.name, md.name AS modname $sectionfield
FROM { course_modules } cm
JOIN { modules } md ON md . id = cm . module
JOIN { " . $modulename . " } m ON m . id = cm . instance
$sectionjoin
WHERE cm . id = : cmid AND md . name = : modulename
$courseselect " ;
2009-07-04 11:15:16 +00:00
return $DB -> get_record_sql ( $sql , $params , $strictness );
2006-08-08 22:09:55 +00:00
}
/**
* Given an instance number of a module , finds the coursemodule description
*
2014-03-14 15:01:40 +08:00
* Please note that this function performs DB query . When possible use cached course
* modinfo . For example get_fast_modinfo ( $courseorid ) -> instances [ $modulename ][ $instance ]
* See also { @ link cm_info :: get_course_module_record ()}
*
2009-05-22 08:41:00 +00:00
* @ global object
2006-08-08 22:09:55 +00:00
* @ param string $modulename name of module type , eg . resource , assignment , ...
* @ param int $instance module instance number ( id in resource , assignment etc . table )
* @ param int $courseid optional course id for extra validation
2008-08-27 11:31:37 +00:00
* @ param bool $sectionnum include relative section number ( 0 , 1 , 2 ... )
2009-07-04 11:15:16 +00:00
* @ param int $strictness IGNORE_MISSING means compatible mode , false returned if record not found , debug message if more found ;
* IGNORE_MULTIPLE means return first , ignore multiple records found ( not recommended );
* MUST_EXIST means throw exception if no record or multiple records found
2010-09-18 13:59:39 +00:00
* @ return stdClass
2004-09-24 21:28:22 +00:00
*/
2009-07-04 11:15:16 +00:00
function get_coursemodule_from_instance ( $modulename , $instance , $courseid = 0 , $sectionnum = false , $strictness = IGNORE_MISSING ) {
2008-05-30 16:47:21 +00:00
global $DB ;
2002-12-17 04:41:18 +00:00
2014-08-07 11:13:22 +12:00
if ( ! core_component :: is_valid_plugin_name ( 'mod' , $modulename )) {
throw new coding_exception ( 'Invalid modulename parameter' );
}
2008-08-27 11:31:37 +00:00
$params = array ( 'instance' => $instance , 'modulename' => $modulename );
2008-05-30 16:51:01 +00:00
$courseselect = " " ;
2008-08-27 11:31:37 +00:00
$sectionfield = " " ;
$sectionjoin = " " ;
2002-12-17 04:41:18 +00:00
2008-05-30 16:47:21 +00:00
if ( $courseid ) {
2008-08-27 11:31:37 +00:00
$courseselect = " AND cm.course = :courseid " ;
2008-05-30 16:47:21 +00:00
$params [ 'courseid' ] = $courseid ;
2008-05-30 16:51:01 +00:00
}
2008-05-30 16:47:21 +00:00
2008-08-27 11:31:37 +00:00
if ( $sectionnum ) {
$sectionfield = " , cw.section AS sectionnum " ;
$sectionjoin = " LEFT JOIN { course_sections} cw ON cw.id = cm.section " ;
}
$sql = " SELECT cm.*, m.name, md.name AS modname $sectionfield
FROM { course_modules } cm
JOIN { modules } md ON md . id = cm . module
JOIN { " . $modulename . " } m ON m . id = cm . instance
$sectionjoin
WHERE m . id = : instance AND md . name = : modulename
$courseselect " ;
2002-12-17 04:41:18 +00:00
2009-07-04 11:15:16 +00:00
return $DB -> get_record_sql ( $sql , $params , $strictness );
2002-12-17 04:41:18 +00:00
}
2008-01-24 20:33:50 +00:00
/**
* Returns all course modules of given activity in course
2009-05-22 08:41:00 +00:00
*
* @ param string $modulename The module name ( forum , quiz , etc . )
* @ param int $courseid The course id to get modules for
2008-01-24 20:33:50 +00:00
* @ param string $extrafields extra fields starting with m .
2009-05-22 08:41:00 +00:00
* @ return array Array of results
2008-01-24 20:33:50 +00:00
*/
function get_coursemodules_in_course ( $modulename , $courseid , $extrafields = '' ) {
2008-05-30 16:47:21 +00:00
global $DB ;
2008-01-24 20:33:50 +00:00
2014-08-07 11:13:22 +12:00
if ( ! core_component :: is_valid_plugin_name ( 'mod' , $modulename )) {
throw new coding_exception ( 'Invalid modulename parameter' );
}
2008-01-24 20:33:50 +00:00
if ( ! empty ( $extrafields )) {
$extrafields = " , $extrafields " ;
}
2008-05-30 16:47:21 +00:00
$params = array ();
$params [ 'courseid' ] = $courseid ;
$params [ 'modulename' ] = $modulename ;
return $DB -> get_records_sql ( " SELECT cm.*, m.name, md.name as modname $extrafields
FROM { course_modules } cm , { modules } md , { " . $modulename . " } m
WHERE cm . course = : courseid AND
cm . instance = m . id AND
md . name = : modulename AND
2008-06-02 10:41:59 +00:00
md . id = cm . module " , $params );
2008-01-24 20:33:50 +00:00
}
2008-02-05 21:34:58 +00:00
2006-01-17 20:49:43 +00:00
/**
* Returns an array of all the active instances of a particular module in given courses , sorted in the order they are defined
*
* Returns an array of all the active instances of a particular
* module in given courses , sorted in the order they are defined
2008-02-05 21:34:58 +00:00
* in the course . Returns an empty array on any errors .
2006-01-17 20:49:43 +00:00
*
2008-02-05 21:34:58 +00:00
* The returned objects includle the columns cw . section , cm . visible ,
2014-08-01 15:05:56 +01:00
* cm . groupmode , and cm . groupingid , and are indexed by cm . id .
2008-02-05 21:34:58 +00:00
*
2009-05-22 08:41:00 +00:00
* @ global object
* @ global object
2008-02-05 21:34:58 +00:00
* @ param string $modulename The name of the module to get instances for
* @ param array $courses an array of course objects .
2009-05-22 08:41:00 +00:00
* @ param int $userid
* @ param int $includeinvisible
2008-02-05 21:34:58 +00:00
* @ return array of module instance objects , including some extra fields from the course_modules
* and course_sections tables , or an empty array if an error occurred .
2006-01-17 20:49:43 +00:00
*/
2006-10-24 20:22:30 +00:00
function get_all_instances_in_courses ( $modulename , $courses , $userid = NULL , $includeinvisible = false ) {
2008-05-30 16:47:21 +00:00
global $CFG , $DB ;
2008-02-05 21:34:58 +00:00
2014-08-07 11:13:22 +12:00
if ( ! core_component :: is_valid_plugin_name ( 'mod' , $modulename )) {
throw new coding_exception ( 'Invalid modulename parameter' );
}
2008-02-05 21:34:58 +00:00
$outputarray = array ();
2006-01-17 20:49:43 +00:00
if ( empty ( $courses ) || ! is_array ( $courses ) || count ( $courses ) == 0 ) {
2008-02-05 21:34:58 +00:00
return $outputarray ;
2006-01-17 20:49:43 +00:00
}
2008-02-05 21:34:58 +00:00
2008-05-30 16:47:21 +00:00
list ( $coursessql , $params ) = $DB -> get_in_or_equal ( array_keys ( $courses ), SQL_PARAMS_NAMED , 'c0' );
$params [ 'modulename' ] = $modulename ;
if ( ! $rawmods = $DB -> get_records_sql ( " SELECT cm.id AS coursemodule, m.*, cw.section, cm.visible AS visible,
2014-08-01 15:05:56 +01:00
cm . groupmode , cm . groupingid
2008-05-30 16:47:21 +00:00
FROM { course_modules } cm , { course_sections } cw , { modules } md ,
{ " . $modulename . " } m
WHERE cm . course $coursessql AND
cm . instance = m . id AND
cm . section = cw . id AND
md . name = : modulename AND
md . id = cm . module " , $params )) {
2008-02-05 21:34:58 +00:00
return $outputarray ;
2006-01-17 20:49:43 +00:00
}
foreach ( $courses as $course ) {
2008-02-05 21:34:58 +00:00
$modinfo = get_fast_modinfo ( $course , $userid );
2006-09-04 21:12:37 +00:00
2008-02-05 21:34:58 +00:00
if ( empty ( $modinfo -> instances [ $modulename ])) {
2006-01-17 20:49:43 +00:00
continue ;
}
2008-02-05 21:34:58 +00:00
foreach ( $modinfo -> instances [ $modulename ] as $cm ) {
if ( ! $includeinvisible and ! $cm -> uservisible ) {
continue ;
}
if ( ! isset ( $rawmods [ $cm -> id ])) {
continue ;
2006-01-17 20:49:43 +00:00
}
2008-02-05 21:34:58 +00:00
$instance = $rawmods [ $cm -> id ];
if ( ! empty ( $cm -> extra )) {
2010-02-14 20:18:10 +00:00
$instance -> extra = $cm -> extra ;
2008-02-05 21:34:58 +00:00
}
$outputarray [] = $instance ;
2006-01-17 20:49:43 +00:00
}
}
return $outputarray ;
}
2004-09-24 21:28:22 +00:00
2003-09-14 04:04:15 +00:00
/**
2007-12-12 17:10:48 +00:00
* Returns an array of all the active instances of a particular module in a given course ,
* sorted in the order they are defined .
2004-09-24 21:28:22 +00:00
*
* Returns an array of all the active instances of a particular
* module in a given course , sorted in the order they are defined
2007-12-12 17:10:48 +00:00
* in the course . Returns an empty array on any errors .
*
* The returned objects includle the columns cw . section , cm . visible ,
2014-08-01 15:05:56 +01:00
* cm . groupmode , and cm . groupingid , and are indexed by cm . id .
2004-09-24 21:28:22 +00:00
*
2009-05-22 08:41:00 +00:00
* Simply calls { @ link all_instances_in_courses ()} with a single provided course
*
2007-12-12 17:10:48 +00:00
* @ param string $modulename The name of the module to get instances for
2008-02-05 21:34:58 +00:00
* @ param object $course The course obect .
2007-12-12 17:10:48 +00:00
* @ return array of module instance objects , including some extra fields from the course_modules
* and course_sections tables , or an empty array if an error occurred .
2009-05-22 08:41:00 +00:00
* @ param int $userid
* @ param int $includeinvisible
2004-09-24 21:28:22 +00:00
*/
2006-10-24 20:22:30 +00:00
function get_all_instances_in_course ( $modulename , $course , $userid = NULL , $includeinvisible = false ) {
2008-02-05 21:34:58 +00:00
return get_all_instances_in_courses ( $modulename , array ( $course -> id => $course ), $userid , $includeinvisible );
2002-12-17 04:41:18 +00:00
}
2002-12-20 14:44:14 +00:00
2003-09-14 04:04:15 +00:00
/**
2004-09-24 21:28:22 +00:00
* Determine whether a module instance is visible within a course
*
* Given a valid module object with info about the id and course ,
* and the module ' s type ( eg " forum " ) returns whether the object
2014-08-01 15:05:56 +01:00
* is visible or not according to the 'eye' icon only .
2004-09-24 21:28:22 +00:00
*
2014-03-26 12:01:52 +00:00
* NOTE : This does NOT take into account visibility to a particular user .
* To get visibility access for a specific user , use get_fast_modinfo , get a
* cm_info object from this , and check the -> uservisible property ; or use
* the \core_availability\info_module :: is_user_visible () static function .
*
2009-05-22 08:41:00 +00:00
* @ global object
2009-11-01 11:31:16 +00:00
2006-09-10 07:07:52 +00:00
* @ param $moduletype Name of the module eg 'forum'
* @ param $module Object which is the instance of the module
2009-05-22 08:41:00 +00:00
* @ return bool Success
2004-09-24 21:28:22 +00:00
*/
2003-04-25 05:24:29 +00:00
function instance_is_visible ( $moduletype , $module ) {
2008-05-30 16:47:21 +00:00
global $DB ;
2003-04-25 05:24:29 +00:00
2004-11-18 02:37:52 +00:00
if ( ! empty ( $module -> id )) {
2008-05-30 16:51:01 +00:00
$params = array ( 'courseid' => $module -> course , 'moduletype' => $moduletype , 'moduleid' => $module -> id );
2014-08-01 15:05:56 +01:00
if ( $records = $DB -> get_records_sql ( " SELECT cm.instance, cm.visible, cm.groupingid, cm.id, cm.course
2008-05-30 16:47:21 +00:00
FROM { course_modules } cm , { modules } m
WHERE cm . course = : courseid AND
cm . module = m . id AND
m . name = : moduletype AND
2008-06-04 07:54:58 +00:00
cm . instance = : moduleid " , $params )) {
2007-08-02 23:39:28 +00:00
2004-11-18 02:37:52 +00:00
foreach ( $records as $record ) { // there should only be one - use the first one
2008-01-24 20:33:50 +00:00
return $record -> visible ;
2004-11-18 02:37:52 +00:00
}
2003-04-25 05:24:29 +00:00
}
}
return true ; // visible by default!
}
2003-01-03 15:31:30 +00:00
2002-12-20 14:44:14 +00:00
/// LOG FUNCTIONS /////////////////////////////////////////////////////
2014-01-13 09:08:58 +08:00
/**
* Get instance of log manager .
*
* @ param bool $forcereload
* @ return \core\log\manager
*/
function get_log_manager ( $forcereload = false ) {
/** @var \core\log\manager $singleton */
static $singleton = null ;
if ( $forcereload and isset ( $singleton )) {
$singleton -> dispose ();
$singleton = null ;
}
if ( isset ( $singleton )) {
return $singleton ;
}
$classname = '\tool_log\log\manager' ;
if ( defined ( 'LOG_MANAGER_CLASS' )) {
$classname = LOG_MANAGER_CLASS ;
}
if ( ! class_exists ( $classname )) {
if ( ! empty ( $classname )) {
debugging ( " Cannot find log manager class ' $classname '. " , DEBUG_DEVELOPER );
}
$classname = '\core\log\dummy_manager' ;
}
$singleton = new $classname ();
return $singleton ;
}
2013-04-12 10:58:59 +08:00
/**
* Add an entry to the config log table .
*
* These are " action " focussed rather than web server hits ,
* and provide a way to easily reconstruct changes to Moodle configuration .
*
* @ package core
* @ category log
* @ global moodle_database $DB
* @ global stdClass $USER
* @ param string $name The name of the configuration change action
For example 'filter_active' when activating or deactivating a filter
* @ param string $oldvalue The config setting ' s previous value
* @ param string $value The config setting ' s new value
* @ param string $plugin Plugin name , for example a filter name when changing filter configuration
* @ return void
*/
function add_to_config_log ( $name , $oldvalue , $value , $plugin ) {
global $USER , $DB ;
$log = new stdClass ();
2017-08-01 13:18:55 +08:00
// Use 0 as user id during install.
$log -> userid = during_initial_install () ? 0 : $USER -> id ;
2013-04-12 10:58:59 +08:00
$log -> timemodified = time ();
$log -> name = $name ;
$log -> oldvalue = $oldvalue ;
$log -> value = $value ;
$log -> plugin = $plugin ;
2017-08-01 13:18:55 +08:00
$id = $DB -> insert_record ( 'config_log' , $log );
$event = core\event\config_log_created :: create ( array (
'objectid' => $id ,
'userid' => $log -> userid ,
'context' => \context_system :: instance (),
'other' => array (
'name' => $log -> name ,
'oldvalue' => $log -> oldvalue ,
'value' => $log -> value ,
'plugin' => $log -> plugin
)
));
$event -> trigger ();
2013-04-12 10:58:59 +08:00
}
2002-12-20 14:44:14 +00:00
2008-04-15 21:46:04 +00:00
/**
* Store user last access times - called when use enters a course or site
*
2012-01-06 10:30:24 +05:30
* @ package core
* @ category log
* @ global stdClass $USER
* @ global stdClass $CFG
* @ global moodle_database $DB
2009-05-22 08:41:00 +00:00
* @ uses LASTACCESS_UPDATE_SECS
* @ uses SITEID
2012-01-05 07:55:47 +05:30
* @ param int $courseid empty courseid means site
2008-04-15 21:46:04 +00:00
* @ return void
*/
function user_accesstime_log ( $courseid = 0 ) {
2008-05-15 21:40:00 +00:00
global $USER , $CFG , $DB ;
2008-04-15 21:46:04 +00:00
2013-09-08 08:38:52 +02:00
if ( ! isloggedin () or \core\session\manager :: is_loggedinas ()) {
2008-04-15 21:46:04 +00:00
// no access tracking
return ;
}
2012-09-05 10:34:51 +08:00
if ( isguestuser ()) {
// Do not update guest access times/ips for performance.
return ;
}
2008-04-15 21:46:04 +00:00
if ( empty ( $courseid )) {
$courseid = SITEID ;
}
$timenow = time ();
/// Store site lastaccess time for the current user
if ( $timenow - $USER -> lastaccess > LASTACCESS_UPDATE_SECS ) {
/// Update $USER->lastaccess for next checks
$USER -> lastaccess = $timenow ;
2010-09-21 08:07:44 +00:00
$last = new stdClass ();
2008-05-15 21:40:00 +00:00
$last -> id = $USER -> id ;
$last -> lastip = getremoteaddr ();
$last -> lastaccess = $timenow ;
2009-02-17 17:15:47 +00:00
$DB -> update_record_raw ( 'user' , $last );
2008-04-15 21:46:04 +00:00
}
if ( $courseid == SITEID ) {
/// no user_lastaccess for frontpage
return ;
}
2008-04-08 23:17:20 +00:00
2008-04-15 21:46:04 +00:00
/// Store course lastaccess times for the current user
if ( empty ( $USER -> currentcourseaccess [ $courseid ]) or ( $timenow - $USER -> currentcourseaccess [ $courseid ] > LASTACCESS_UPDATE_SECS )) {
2008-05-15 21:40:00 +00:00
$lastaccess = $DB -> get_field ( 'user_lastaccess' , 'timeaccess' , array ( 'userid' => $USER -> id , 'courseid' => $courseid ));
2008-04-15 21:46:04 +00:00
2008-05-15 21:40:00 +00:00
if ( $lastaccess === false ) {
// Update course lastaccess for next checks
$USER -> currentcourseaccess [ $courseid ] = $timenow ;
2010-09-21 08:07:44 +00:00
$last = new stdClass ();
2008-05-15 21:40:00 +00:00
$last -> userid = $USER -> id ;
$last -> courseid = $courseid ;
$last -> timeaccess = $timenow ;
2013-12-12 03:33:20 +11:00
try {
$DB -> insert_record_raw ( 'user_lastaccess' , $last , false );
} catch ( dml_write_exception $e ) {
// During a race condition we can fail to find the data, then it appears.
// If we still can't find it, rethrow the exception.
$lastaccess = $DB -> get_field ( 'user_lastaccess' , 'timeaccess' , array ( 'userid' => $USER -> id ,
'courseid' => $courseid ));
if ( $lastaccess === false ) {
throw $e ;
}
// If we did find it, the race condition was true and another thread has inserted the time for us.
// We can just continue without having to do anything.
}
2008-05-30 16:51:01 +00:00
2008-05-15 21:40:00 +00:00
} else if ( $timenow - $lastaccess < LASTACCESS_UPDATE_SECS ) {
// no need to update now, it was updated recently in concurrent login ;-)
2008-04-15 21:46:04 +00:00
2008-05-15 21:40:00 +00:00
} else {
// Update course lastaccess for next checks
$USER -> currentcourseaccess [ $courseid ] = $timenow ;
2009-02-17 17:15:47 +00:00
$DB -> set_field ( 'user_lastaccess' , 'timeaccess' , $timenow , array ( 'userid' => $USER -> id , 'courseid' => $courseid ));
2004-01-30 18:21:56 +00:00
}
2004-09-21 11:41:58 +00:00
}
2002-12-20 14:44:14 +00:00
}
2003-01-03 15:31:30 +00:00
/// GENERAL HELPFUL THINGS ///////////////////////////////////
2003-09-14 04:04:15 +00:00
/**
2012-01-03 11:29:10 +01:00
* Dumps a given object ' s information for debugging purposes
2004-09-24 21:28:22 +00:00
*
2012-01-03 11:29:10 +01:00
* When used in a CLI script , the object ' s information is written to the standard
* error output stream . When used in a web script , the object is dumped to a
* pre - formatted block with the " notifytiny " CSS class .
2004-09-24 21:28:22 +00:00
*
* @ param mixed $object The data to be printed
2012-01-03 11:29:10 +01:00
* @ return void output is echo ' d
2004-09-24 21:28:22 +00:00
*/
2003-01-03 15:31:30 +00:00
function print_object ( $object ) {
2012-01-03 11:29:10 +01:00
2011-11-07 15:18:32 +08:00
// we may need a lot of memory here
raise_memory_limit ( MEMORY_EXTRA );
2012-01-03 11:29:10 +01:00
if ( CLI_SCRIPT ) {
fwrite ( STDERR , print_r ( $object , true ));
fwrite ( STDERR , PHP_EOL );
2019-10-03 17:06:59 +08:00
} else if ( AJAX_SCRIPT ) {
foreach ( explode ( " \n " , print_r ( $object , true )) as $line ) {
error_log ( $line );
}
2012-01-03 11:29:10 +01:00
} else {
echo html_writer :: tag ( 'pre' , s ( print_r ( $object , true )), array ( 'class' => 'notifytiny' ));
}
2003-01-03 15:31:30 +00:00
}
2006-10-28 15:20:14 +00:00
/**
2007-08-02 23:39:28 +00:00
* This function is the official hook inside XMLDB stuff to delegate its debug to one
2006-10-28 15:20:14 +00:00
* external function .
*
* Any script can avoid calls to this function by defining XMLDB_SKIP_DEBUG_HOOK before
* using XMLDB classes . Obviously , also , if this function doesn 't exist, it isn' t invoked ; - )
*
2009-05-22 08:41:00 +00:00
* @ uses DEBUG_DEVELOPER
* @ param string $message string contains the error message
* @ param object $object object XMLDB object that fired the debug
2006-10-28 15:20:14 +00:00
*/
function xmldb_debug ( $message , $object ) {
2007-08-12 15:58:08 +00:00
debugging ( $message , DEBUG_DEVELOPER );
2006-10-28 15:20:14 +00:00
}
2007-04-03 09:19:09 +00:00
/**
2009-05-22 08:41:00 +00:00
* @ global object
* @ uses CONTEXT_COURSECAT
2008-12-01 06:55:11 +00:00
* @ return boolean Whether the user can create courses in any category in the system .
2007-04-03 09:19:09 +00:00
*/
function user_can_create_courses () {
2008-05-30 16:47:21 +00:00
global $DB ;
2008-12-01 06:55:11 +00:00
$catsrs = $DB -> get_recordset ( 'course_categories' );
2009-01-11 17:16:50 +00:00
foreach ( $catsrs as $cat ) {
2012-07-25 16:25:55 +08:00
if ( has_capability ( 'moodle/course:create' , context_coursecat :: instance ( $cat -> id ))) {
2008-12-01 06:55:11 +00:00
$catsrs -> close ();
return true ;
2007-04-03 09:19:09 +00:00
}
}
2008-12-01 06:55:11 +00:00
$catsrs -> close ();
return false ;
2007-04-03 09:19:09 +00:00
}
2014-02-17 16:48:27 +00:00
/**
* This method can update the values in mulitple database rows for a colum with
* a unique index , without violating that constraint .
*
* Suppose we have a table with a unique index on ( otherid , sortorder ), and
* for a particular value of otherid , we want to change all the sort orders .
* You have to do this carefully or you will violate the unique index at some time .
* This method takes care of the details for you .
*
* Note that , it is the responsibility of the caller to make sure that the
* requested rename is legal . For example , if you ask for [ 1 => 2 , 2 => 2 ]
* then you will get a unique key violation error from the database .
*
* @ param string $table The database table to modify .
* @ param string $field the field that contains the values we are going to change .
* @ param array $newvalues oldvalue => newvalue how to change the values .
* E . g . [ 1 => 4 , 2 => 1 , 3 => 3 , 4 => 2 ] .
* @ param array $otherconditions array fieldname => requestedvalue extra WHERE clause
* conditions to restrict which rows are affected . E . g . array ( 'otherid' => 123 ) .
* @ param int $unusedvalue ( defaults to - 1 ) a value that is never used in $ordercol .
*/
function update_field_with_unique_index ( $table , $field , array $newvalues ,
array $otherconditions , $unusedvalue = - 1 ) {
global $DB ;
$safechanges = decompose_update_into_safe_changes ( $newvalues , $unusedvalue );
$transaction = $DB -> start_delegated_transaction ();
foreach ( $safechanges as $change ) {
list ( $from , $to ) = $change ;
$otherconditions [ $field ] = $from ;
$DB -> set_field ( $table , $field , $to , $otherconditions );
}
$transaction -> allow_commit ();
}
/**
* Helper used by { @ link update_field_with_unique_index ()} . Given a desired
* set of changes , break them down into single udpates that can be done one at
* a time without breaking any unique index constraints .
*
* Suppose the input is array ( 1 => 2 , 2 => 1 ) and - 1. Then the output will be
* array ( array ( 1 , - 1 ), array ( 2 , 1 ), array ( - 1 , 2 )) . This function solves this
* problem in the general case , not just for simple swaps . The unit tests give
* more examples .
*
* Note that , it is the responsibility of the caller to make sure that the
* requested rename is legal . For example , if you ask for something impossible
* like array ( 1 => 2 , 2 => 2 ) then the results are undefined . ( You will probably
* get a unique key violation error from the database later . )
*
* @ param array $newvalues The desired re - ordering .
* E . g . array ( 1 => 4 , 2 => 1 , 3 => 3 , 4 => 2 ) .
* @ param int $unusedvalue A value that is not currently used .
* @ return array A safe way to perform the re - order . An array of two - element
* arrays array ( $from , $to ) .
* E . g . array ( array ( 1 , - 1 ), array ( 2 , 1 ), array ( 4 , 2 ), array ( - 1 , 4 )) .
*/
function decompose_update_into_safe_changes ( array $newvalues , $unusedvalue ) {
$nontrivialmap = array ();
foreach ( $newvalues as $from => $to ) {
if ( $from == $unusedvalue || $to == $unusedvalue ) {
throw new \coding_exception ( 'Supposedly unused value ' . $unusedvalue . ' is actually used!' );
}
if ( $from != $to ) {
$nontrivialmap [ $from ] = $to ;
}
}
if ( empty ( $nontrivialmap )) {
return array ();
}
// First we deal with all renames that are not part of cycles.
// This bit is O(n^2) and it ought to be possible to do better,
// but it does not seem worth the effort.
$safechanges = array ();
$nontrivialmapchanged = true ;
while ( $nontrivialmapchanged ) {
$nontrivialmapchanged = false ;
foreach ( $nontrivialmap as $from => $to ) {
if ( array_key_exists ( $to , $nontrivialmap )) {
continue ; // Cannot currenly do this rename.
}
// Is safe to do this rename now.
$safechanges [] = array ( $from , $to );
unset ( $nontrivialmap [ $from ]);
$nontrivialmapchanged = true ;
}
}
// Are we done?
if ( empty ( $nontrivialmap )) {
return $safechanges ;
}
// Now what is left in $nontrivialmap must be a permutation,
// which must be a combination of disjoint cycles. We need to break them.
while ( ! empty ( $nontrivialmap )) {
// Extract the first cycle.
reset ( $nontrivialmap );
$current = $cyclestart = key ( $nontrivialmap );
$cycle = array ();
do {
$cycle [] = $current ;
$next = $nontrivialmap [ $current ];
unset ( $nontrivialmap [ $current ]);
$current = $next ;
2014-03-18 17:55:58 +00:00
} while ( $current != $cyclestart );
2014-02-17 16:48:27 +00:00
// Now convert it to a sequence of safe renames by using a temp.
$safechanges [] = array ( $cyclestart , $unusedvalue );
$cycle [ 0 ] = $unusedvalue ;
$to = $cyclestart ;
while ( $from = array_pop ( $cycle )) {
$safechanges [] = array ( $from , $to );
$to = $from ;
}
}
return $safechanges ;
}