2009-05-19 07:50:54 +00:00
< ? php
2009-11-01 11:31:16 +00:00
// This file is part of Moodle - http://moodle.org/
//
2009-05-19 07:50:54 +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-19 07:50:54 +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/>.
2007-08-06 17:55:44 +00:00
2007-09-19 07:15:27 +00:00
/**
2009-05-19 07:50:54 +00:00
* This file contains functions for managing user access
*
2009-05-22 02:03:55 +00:00
* < b > Public API vs internals </ b >
2008-05-24 18:35:48 +00:00
*
2007-09-19 07:15:27 +00:00
* General users probably only care about
*
2007-09-19 07:57:10 +00:00
* Context handling
2011-10-14 12:48:00 +02:00
* - context_course :: instance ( $courseid ), context_module :: instance ( $cm -> id ), context_coursecat :: instance ( $catid )
* - context :: instance_by_id ( $contextid )
* - $context -> get_parent_contexts ();
* - $context -> get_child_contexts ();
2008-05-24 18:35:48 +00:00
*
2007-09-19 07:57:10 +00:00
* Whether the user can do something ...
2007-09-19 07:15:27 +00:00
* - has_capability ()
2008-12-08 07:28:19 +00:00
* - has_any_capability ()
* - has_all_capabilities ()
2007-09-19 07:24:43 +00:00
* - require_capability ()
2007-09-19 07:57:10 +00:00
* - require_login () ( from moodlelib )
2012-01-14 19:12:43 +01:00
* - is_enrolled ()
* - is_viewing ()
* - is_guest ()
2011-10-14 12:48:00 +02:00
* - is_siteadmin ()
2012-01-14 19:12:43 +01:00
* - isguestuser ()
* - isloggedin ()
2007-09-19 07:57:10 +00:00
*
* What courses has this user access to ?
2011-10-14 12:48:00 +02:00
* - get_enrolled_users ()
2007-09-19 07:57:10 +00:00
*
2008-04-29 06:53:19 +00:00
* What users can do X in this context ?
2012-01-14 19:12:43 +01:00
* - get_enrolled_users () - at and bellow course context
* - get_users_by_capability () - above course context
2008-04-29 06:53:19 +00:00
*
2011-10-14 12:48:00 +02:00
* Modify roles
* - role_assign ()
* - role_unassign ()
* - role_unassign_all ()
2008-05-24 18:35:48 +00:00
*
2011-10-14 12:48:00 +02:00
* Advanced - for internal use only
2007-09-19 07:57:10 +00:00
* - load_all_capabilities ()
* - reload_all_capabilities ()
2007-09-20 04:52:54 +00:00
* - has_capability_in_accessdata ()
2007-09-19 07:57:10 +00:00
* - get_user_access_sitewide ()
2011-10-14 12:48:00 +02:00
* - load_course_context ()
* - load_role_access_by_context ()
* - etc .
2007-09-19 07:57:10 +00:00
*
2009-05-22 02:03:55 +00:00
* < b > Name conventions </ b >
2008-05-24 18:35:48 +00:00
*
2009-05-22 02:03:55 +00:00
* " ctx " means context
2007-09-19 07:15:27 +00:00
*
2009-05-22 02:03:55 +00:00
* < b > accessdata </ b >
2007-09-19 07:15:27 +00:00
*
* Access control data is held in the " accessdata " array
* which - for the logged - in user , will be in $USER -> access
2008-05-24 18:35:48 +00:00
*
2009-03-23 03:54:50 +00:00
* For other users can be generated and passed around ( but may also be cached
2011-10-14 12:48:00 +02:00
* against userid in $ACCESSLIB_PRIVATE -> accessdatabyuser ) .
2007-09-19 07:15:27 +00:00
*
2007-09-20 04:52:54 +00:00
* $accessdata is a multidimensional array , holding
2008-05-24 18:35:48 +00:00
* role assignments ( RAs ), role - capabilities - perm sets
2007-09-19 07:15:50 +00:00
* ( role defs ) and a list of courses we have loaded
2007-09-19 07:15:27 +00:00
* data for .
*
2008-05-24 18:35:48 +00:00
* Things are keyed on " contextpaths " ( the path field of
2007-09-19 07:15:27 +00:00
* the context table ) for fast walking up / down the tree .
2009-05-22 02:03:55 +00:00
* < code >
2011-10-14 12:48:00 +02:00
* $accessdata [ 'ra' ][ $contextpath ] = array ( $roleid => $roleid )
* [ $contextpath ] = array ( $roleid => $roleid )
* [ $contextpath ] = array ( $roleid => $roleid )
2009-11-01 11:31:16 +00:00
* </ code >
2007-09-19 07:15:27 +00:00
*
* Role definitions are stored like this
* ( no cap merge is done - so it ' s compact )
*
2009-05-22 02:03:55 +00:00
* < code >
2011-10-14 12:48:00 +02:00
* $accessdata [ 'rdef' ][ " $contextpath : $roleid " ][ 'mod/forum:viewpost' ] = 1
* [ 'mod/forum:editallpost' ] = - 1
* [ 'mod/forum:startdiscussion' ] = - 1000
2009-05-22 02:03:55 +00:00
* </ code >
2007-09-19 07:15:27 +00:00
*
2011-10-14 12:48:00 +02:00
* See how has_capability_in_accessdata () walks up the tree .
2007-09-19 07:15:27 +00:00
*
2011-10-14 12:48:00 +02:00
* First we only load rdef and ra down to the course level , but not below .
* This keeps accessdata small and compact . Below - the - course ra / rdef
* are loaded as needed . We keep track of which courses we have loaded ra / rdef in
2009-05-22 02:03:55 +00:00
* < code >
2011-10-14 12:48:00 +02:00
* $accessdata [ 'loaded' ] = array ( $courseid1 => 1 , $courseid2 => 1 )
2009-05-22 02:03:55 +00:00
* </ code >
2007-09-19 07:15:27 +00:00
*
2009-05-22 02:03:55 +00:00
* < b > Stale accessdata </ b >
2007-09-19 07:15:27 +00:00
*
* For the logged - in user , accessdata is long - lived .
*
2009-03-23 03:54:50 +00:00
* On each pageload we load $ACCESSLIB_PRIVATE -> dirtycontexts which lists
2007-09-19 07:15:27 +00:00
* context paths affected by changes . Any check at - or - below
* a dirty context will trigger a transparent reload of accessdata .
2008-05-24 18:35:48 +00:00
*
2010-05-22 18:46:43 +00:00
* Changes at the system level will force the reload for everyone .
2007-09-19 07:15:27 +00:00
*
2009-05-22 02:03:55 +00:00
* < b > Default role caps </ b >
2008-05-24 18:35:48 +00:00
* The default role assignment is not in the DB , so we
* add it manually to accessdata .
2007-09-19 07:15:27 +00:00
*
* This means that functions that work directly off the
* DB need to ensure that the default role caps
2008-05-24 18:35:48 +00:00
* are dealt with appropriately .
2007-09-19 07:15:27 +00:00
*
2012-01-14 19:12:43 +01:00
* @ package core_access
2010-07-25 13:35:05 +00:00
* @ copyright 1999 onwards Martin Dougiamas http :// dougiamas . com
* @ license http :// www . gnu . org / copyleft / gpl . html GNU GPL v3 or later
2007-09-19 07:15:27 +00:00
*/
2006-08-08 05:13:06 +00:00
2010-07-25 13:35:05 +00:00
defined ( 'MOODLE_INTERNAL' ) || die ();
2011-10-14 12:48:00 +02:00
/** No capability change */
2006-09-15 06:07:29 +00:00
define ( 'CAP_INHERIT' , 0 );
2011-10-14 12:48:00 +02:00
/** Allow permission, overrides CAP_PREVENT defined in parent contexts */
2006-08-08 05:13:06 +00:00
define ( 'CAP_ALLOW' , 1 );
2011-10-14 12:48:00 +02:00
/** Prevent permission, overrides CAP_ALLOW defined in parent contexts */
2006-08-08 05:13:06 +00:00
define ( 'CAP_PREVENT' , - 1 );
2011-10-14 12:48:00 +02:00
/** Prohibit permission, overrides everything in current and child contexts */
2006-08-08 05:13:06 +00:00
define ( 'CAP_PROHIBIT' , - 1000 );
2011-10-14 12:48:00 +02:00
/** System context level - only one instance in every system */
2006-08-08 05:13:06 +00:00
define ( 'CONTEXT_SYSTEM' , 10 );
2011-10-14 12:48:00 +02:00
/** User context level - one instance for each user describing what others can do to user */
2006-09-15 14:09:16 +00:00
define ( 'CONTEXT_USER' , 30 );
2011-10-14 12:48:00 +02:00
/** Course category context level - one instance for each category */
2006-08-08 05:13:06 +00:00
define ( 'CONTEXT_COURSECAT' , 40 );
2011-10-14 12:48:00 +02:00
/** Course context level - one instances for each course */
2006-08-08 05:13:06 +00:00
define ( 'CONTEXT_COURSE' , 50 );
2011-10-14 12:48:00 +02:00
/** Course module context level - one instance for each course module */
2006-08-08 05:13:06 +00:00
define ( 'CONTEXT_MODULE' , 70 );
2011-10-14 12:48:00 +02:00
/**
* Block context level - one instance for each block , sticky blocks are tricky
* because ppl think they should be able to override them at lower contexts .
* Any other context level instance can be parent of block context .
*/
2006-08-08 05:13:06 +00:00
define ( 'CONTEXT_BLOCK' , 80 );
2011-10-14 12:48:00 +02:00
/** Capability allow management of trusts - NOT IMPLEMENTED YET - see {@link http://docs.moodle.org/dev/Hardening_new_Roles_system} */
2006-09-18 21:32:49 +00:00
define ( 'RISK_MANAGETRUST' , 0x0001 );
2011-10-14 12:48:00 +02:00
/** Capability allows changes in system configuration - see {@link http://docs.moodle.org/dev/Hardening_new_Roles_system} */
2006-09-19 07:33:22 +00:00
define ( 'RISK_CONFIG' , 0x0002 );
2012-01-14 19:12:43 +01:00
/** Capability allows user to add scripted content - see {@link http://docs.moodle.org/dev/Hardening_new_Roles_system} */
2006-09-18 21:32:49 +00:00
define ( 'RISK_XSS' , 0x0004 );
2011-10-14 12:48:00 +02:00
/** Capability allows access to personal user information - see {@link http://docs.moodle.org/dev/Hardening_new_Roles_system} */
2006-09-18 21:32:49 +00:00
define ( 'RISK_PERSONAL' , 0x0008 );
2012-01-14 19:12:43 +01:00
/** Capability allows users to add content others may see - see {@link http://docs.moodle.org/dev/Hardening_new_Roles_system} */
2006-09-18 21:32:49 +00:00
define ( 'RISK_SPAM' , 0x0010 );
2011-10-14 12:48:00 +02:00
/** capability allows mass delete of data belonging to other users - see {@link http://docs.moodle.org/dev/Hardening_new_Roles_system} */
2008-07-23 16:10:06 +00:00
define ( 'RISK_DATALOSS' , 0x0020 );
2006-09-18 21:32:49 +00:00
2012-05-18 11:40:35 +02:00
/** rolename displays - the name as defined in the role definition, localised if name empty */
2009-05-22 02:03:55 +00:00
define ( 'ROLENAME_ORIGINAL' , 0 );
2012-05-18 11:40:35 +02:00
/** rolename displays - the name as defined by a role alias at the course level, falls back to ROLENAME_ORIGINAL if alias not present */
2009-05-22 02:03:55 +00:00
define ( 'ROLENAME_ALIAS' , 1 );
2011-10-14 12:48:00 +02:00
/** rolename displays - Both, like this: Role alias (Original) */
2009-05-22 02:03:55 +00:00
define ( 'ROLENAME_BOTH' , 2 );
2011-10-14 12:48:00 +02:00
/** rolename displays - the name as defined in the role definition and the shortname in brackets */
2009-05-22 02:03:55 +00:00
define ( 'ROLENAME_ORIGINALANDSHORT' , 3 );
2011-10-14 12:48:00 +02:00
/** rolename displays - the name as defined by a role alias, in raw form suitable for editing */
2009-05-22 02:03:55 +00:00
define ( 'ROLENAME_ALIAS_RAW' , 4 );
2011-10-14 12:48:00 +02:00
/** rolename displays - the name is simply short role name */
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
define ( 'ROLENAME_SHORT' , 5 );
2009-05-22 02:03:55 +00:00
2011-10-14 12:48:00 +02:00
if ( ! defined ( 'CONTEXT_CACHE_MAX_SIZE' )) {
2012-01-14 19:12:43 +01:00
/** maximum size of context cache - it is possible to tweak this config.php or in any script before inclusion of context.php */
2011-10-14 12:48:00 +02:00
define ( 'CONTEXT_CACHE_MAX_SIZE' , 2500 );
2009-04-06 09:48:39 +00:00
}
2009-05-22 02:03:55 +00:00
/**
2009-11-01 11:31:16 +00:00
* Although this looks like a global variable , it isn ' t really .
2009-05-22 02:03:55 +00:00
*
2009-11-01 11:31:16 +00:00
* It is just a private implementation detail to accesslib that MUST NOT be used elsewhere .
* It is used to cache various bits of data between function calls for performance reasons .
2010-05-22 18:46:43 +00:00
* Sadly , a PHP global variable is the only way to implement this , without rewriting everything
2009-05-22 02:03:55 +00:00
* as methods of a class , instead of functions .
*
2012-01-14 19:12:43 +01:00
* @ access private
2009-05-22 02:03:55 +00:00
* @ global stdClass $ACCESSLIB_PRIVATE
* @ name $ACCESSLIB_PRIVATE
*/
2010-08-26 16:10:19 +00:00
global $ACCESSLIB_PRIVATE ;
2010-10-01 21:45:14 +00:00
$ACCESSLIB_PRIVATE = new stdClass ();
2011-10-14 12:48:00 +02:00
$ACCESSLIB_PRIVATE -> dirtycontexts = null ; // Dirty contexts cache, loaded from DB once per page
$ACCESSLIB_PRIVATE -> accessdatabyuser = array (); // Holds the cache of $accessdata structure for users (including $USER)
$ACCESSLIB_PRIVATE -> rolepermissions = array (); // role permissions cache - helps a lot with mem usage
$ACCESSLIB_PRIVATE -> capabilities = null ; // detailed information about the capabilities
2006-08-08 05:13:06 +00:00
2009-03-23 03:54:50 +00:00
/**
2009-05-19 07:50:54 +00:00
* Clears accesslib ' s private caches . ONLY BE USED BY UNIT TESTS
2009-11-01 11:31:16 +00:00
*
2009-03-23 03:54:50 +00:00
* This method should ONLY BE USED BY UNIT TESTS . It clears all of
* accesslib ' s private caches . You need to do this before setting up test data ,
2010-05-22 18:46:43 +00:00
* and also at the end of the tests .
2011-10-14 12:48:00 +02:00
*
2012-01-14 19:12:43 +01:00
* @ access private
2011-10-14 12:48:00 +02:00
* @ return void
2009-03-23 03:54:50 +00:00
*/
function accesslib_clear_all_caches_for_unit_testing () {
2012-07-21 10:11:50 +02:00
global $USER ;
if ( ! PHPUNIT_TEST ) {
2009-03-23 03:54:50 +00:00
throw new coding_exception ( 'You must not call clear_all_caches outside of unit tests.' );
}
2011-10-14 12:48:00 +02:00
accesslib_clear_all_caches ( true );
2009-03-23 03:54:50 +00:00
unset ( $USER -> access );
}
2009-05-19 07:50:54 +00:00
/**
2011-10-14 12:48:00 +02:00
* Clears accesslib ' s private caches . ONLY BE USED FROM THIS LIBRARY FILE !
2009-05-19 07:50:54 +00:00
*
2011-10-14 12:48:00 +02:00
* This reset does not touch global $USER .
*
2012-01-14 19:12:43 +01:00
* @ access private
2011-10-14 12:48:00 +02:00
* @ param bool $resetcontexts
* @ return void
2009-05-19 07:50:54 +00:00
*/
2011-10-14 12:48:00 +02:00
function accesslib_clear_all_caches ( $resetcontexts ) {
global $ACCESSLIB_PRIVATE ;
2006-09-12 06:15:33 +00:00
2011-10-14 12:48:00 +02:00
$ACCESSLIB_PRIVATE -> dirtycontexts = null ;
$ACCESSLIB_PRIVATE -> accessdatabyuser = array ();
$ACCESSLIB_PRIVATE -> rolepermissions = array ();
$ACCESSLIB_PRIVATE -> capabilities = null ;
2006-09-12 06:15:33 +00:00
2011-10-14 12:48:00 +02:00
if ( $resetcontexts ) {
context_helper :: reset_caches ();
2006-09-12 06:15:33 +00:00
}
2007-03-07 13:45:41 +00:00
}
/**
2009-05-19 07:50:54 +00:00
* Gets the accessdata for role " sitewide " ( system down to course )
2007-09-19 07:16:29 +00:00
*
2012-01-14 19:12:43 +01:00
* @ access private
2009-05-19 07:50:54 +00:00
* @ param int $roleid
2007-09-19 07:02:57 +00:00
* @ return array
2007-03-07 13:45:41 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_role_access ( $roleid ) {
global $DB , $ACCESSLIB_PRIVATE ;
2007-03-07 13:45:41 +00:00
2011-10-14 12:48:00 +02:00
/* Get it in 1 DB query ...
2007-09-19 07:02:57 +00:00
* - relevant role caps at the root and down
* to the course level - but not below
*/
2011-10-14 12:48:00 +02:00
//TODO: MUC - this could be cached in shared memory to speed up first page loading, web crawlers, etc.
$accessdata = get_empty_accessdata ();
$accessdata [ 'ra' ][ '/' . SYSCONTEXTID ] = array (( int ) $roleid => ( int ) $roleid );
2006-09-12 06:15:33 +00:00
2012-09-28 11:39:41 +02:00
// Overrides for the role IN ANY CONTEXTS down to COURSE - not below -.
/*
2007-09-19 07:02:57 +00:00
$sql = " SELECT ctx.path,
rc . capability , rc . permission
2008-05-15 21:40:00 +00:00
FROM { context } ctx
2011-10-14 12:48:00 +02:00
JOIN { role_capabilities } rc ON rc . contextid = ctx . id
LEFT JOIN { context } cctx
ON ( cctx . contextlevel = " .CONTEXT_COURSE. " AND ctx . path LIKE " . $DB->sql_concat ('cctx.path', " '/%' " ). " )
WHERE rc . roleid = ? AND cctx . id IS NULL " ;
2008-05-15 21:40:00 +00:00
$params = array ( $roleid );
2012-09-28 11:39:41 +02:00
*/
// Note: the commented out query is 100% accurate but slow, so let's cheat instead by hardcoding the blocks mess directly.
$sql = " SELECT COALESCE(ctx.path, bctx.path) AS path, rc.capability, rc.permission
FROM { role_capabilities } rc
LEFT JOIN { context } ctx ON ( ctx . id = rc . contextid AND ctx . contextlevel <= " .CONTEXT_COURSE. " )
LEFT JOIN ({ context } bctx
JOIN { block_instances } bi ON ( bi . id = bctx . instanceid )
JOIN { context } pctx ON ( pctx . id = bi . parentcontextid AND pctx . contextlevel < " .CONTEXT_COURSE. " )
) ON ( bctx . id = rc . contextid AND bctx . contextlevel = " .CONTEXT_BLOCK. " )
WHERE rc . roleid = : roleid AND ( ctx . id IS NOT NULL OR bctx . id IS NOT NULL ) " ;
$params = array ( 'roleid' => $roleid );
2008-02-05 10:41:05 +00:00
2009-01-03 14:28:02 +00:00
// we need extra caching in CLI scripts and cron
2011-10-14 12:48:00 +02:00
$rs = $DB -> get_recordset_sql ( $sql , $params );
foreach ( $rs as $rd ) {
$k = " { $rd -> path } : { $roleid } " ;
$accessdata [ 'rdef' ][ $k ][ $rd -> capability ] = ( int ) $rd -> permission ;
2007-01-28 20:52:57 +00:00
}
2011-10-14 12:48:00 +02:00
$rs -> close ();
2007-09-19 07:02:57 +00:00
2011-10-14 12:48:00 +02:00
// share the role definitions
foreach ( $accessdata [ 'rdef' ] as $k => $unused ) {
if ( ! isset ( $ACCESSLIB_PRIVATE -> rolepermissions [ $k ])) {
$ACCESSLIB_PRIVATE -> rolepermissions [ $k ] = $accessdata [ 'rdef' ][ $k ];
2007-09-26 07:12:38 +00:00
}
2011-10-14 12:48:00 +02:00
$accessdata [ 'rdef_count' ] ++ ;
$accessdata [ 'rdef' ][ $k ] =& $ACCESSLIB_PRIVATE -> rolepermissions [ $k ];
2007-09-26 07:12:38 +00:00
}
return $accessdata ;
}
2006-09-14 09:39:23 +00:00
/**
2011-10-14 12:48:00 +02:00
* Get the default guest role , this is used for guest account ,
* search engine spiders , etc .
2009-11-01 11:31:16 +00:00
*
2011-10-14 12:48:00 +02:00
* @ return stdClass role record
2006-09-14 09:39:23 +00:00
*/
function get_guest_role () {
2008-05-15 21:40:00 +00:00
global $CFG , $DB ;
2007-02-01 10:24:26 +00:00
if ( empty ( $CFG -> guestroleid )) {
2010-03-31 07:41:31 +00:00
if ( $roles = $DB -> get_records ( 'role' , array ( 'archetype' => 'guest' ))) {
2007-02-01 10:24:26 +00:00
$guestrole = array_shift ( $roles ); // Pick the first one
set_config ( 'guestroleid' , $guestrole -> id );
return $guestrole ;
} else {
debugging ( 'Can not find any guest role!' );
return false ;
}
2006-09-14 09:39:23 +00:00
} else {
2008-05-15 21:40:00 +00:00
if ( $guestrole = $DB -> get_record ( 'role' , array ( 'id' => $CFG -> guestroleid ))) {
2007-02-01 10:24:26 +00:00
return $guestrole ;
} else {
2011-10-14 12:48:00 +02:00
// somebody is messing with guest roles, remove incorrect setting and try to find a new one
2007-02-01 10:24:26 +00:00
set_config ( 'guestroleid' , '' );
return get_guest_role ();
}
2006-09-14 09:39:23 +00:00
}
}
2007-10-09 12:49:54 +00:00
/**
2010-05-22 18:46:43 +00:00
* Check whether a user has a particular capability in a given context .
2009-05-19 07:50:54 +00:00
*
2011-10-14 12:48:00 +02:00
* For example :
2012-01-14 19:12:43 +01:00
* $context = context_module :: instance ( $cm -> id );
* has_capability ( 'mod/forum:replypost' , $context )
2009-05-19 07:50:54 +00:00
*
2010-05-22 18:46:43 +00:00
* By default checks the capabilities of the current user , but you can pass a
2010-03-31 07:41:31 +00:00
* different userid . By default will return true for admin users , but you can override that with the fourth argument .
*
* Guest and not - logged - in users can never get any dangerous capability - that is any write capability
* or capabilities with XSS , config or data loss risks .
2009-11-01 11:31:16 +00:00
*
2012-01-14 19:12:43 +01:00
* @ category access
*
2009-07-21 06:24:44 +00:00
* @ param string $capability the name of the capability to check . For example mod / forum : view
2012-01-14 19:12:43 +01:00
* @ param context $context the context to check the capability in . You normally get this with instance method of a context class .
* @ param integer | stdClass $user A user id or object . By default ( null ) checks the permissions of the current user .
2010-03-31 07:41:31 +00:00
* @ param boolean $doanything If false , ignores effect of admin role assignment
2009-07-21 06:24:44 +00:00
* @ return boolean true if the user has this capability . Otherwise false .
2007-10-09 12:49:54 +00:00
*/
2011-10-14 12:48:00 +02:00
function has_capability ( $capability , context $context , $user = null , $doanything = true ) {
global $USER , $CFG , $SCRIPT , $ACCESSLIB_PRIVATE ;
2009-01-15 22:09:57 +00:00
2009-06-24 09:17:56 +00:00
if ( during_initial_install ()) {
2011-10-14 12:48:00 +02:00
if ( $SCRIPT === " / $CFG->admin /index.php " or $SCRIPT === " / $CFG->admin /cli/install.php " or $SCRIPT === " / $CFG->admin /cli/install_database.php " ) {
2009-01-15 22:09:57 +00:00
// we are in an installer - roles can not work yet
return true ;
} else {
return false ;
}
}
2007-09-19 07:02:31 +00:00
2010-03-31 07:41:31 +00:00
if ( strpos ( $capability , 'moodle/legacy:' ) === 0 ) {
throw new coding_exception ( 'Legacy capabilities can not be used any more!' );
}
2011-10-14 12:48:00 +02:00
if ( ! is_bool ( $doanything )) {
throw new coding_exception ( 'Capability parameter "doanything" is wierd, only true or false is allowed. This has to be fixed in code.' );
}
// capability must exist
if ( ! $capinfo = get_capability_info ( $capability )) {
debugging ( 'Capability "' . $capability . '" was not found! This has to be fixed in code.' );
2007-10-05 15:06:38 +00:00
return false ;
2007-09-19 07:02:44 +00:00
}
2011-10-14 12:48:00 +02:00
if ( ! isset ( $USER -> id )) {
// should never happen
$USER -> id = 0 ;
2012-10-26 10:43:39 +08:00
debugging ( 'Capability check being performed on a user with no ID.' , DEBUG_DEVELOPER );
2010-03-31 07:41:31 +00:00
}
2007-09-19 07:02:31 +00:00
2010-03-31 07:41:31 +00:00
// make sure there is a real user specified
2010-10-01 21:45:14 +00:00
if ( $user === null ) {
2011-10-14 12:48:00 +02:00
$userid = $USER -> id ;
2010-03-31 07:41:31 +00:00
} else {
2011-07-24 15:04:17 +02:00
$userid = is_object ( $user ) ? $user -> id : $user ;
2008-01-08 15:04:00 +00:00
}
2011-10-14 12:48:00 +02:00
// make sure forcelogin cuts off not-logged-in users if enabled
if ( ! empty ( $CFG -> forcelogin ) and $userid == 0 ) {
2010-03-31 07:41:31 +00:00
return false ;
}
2011-10-14 12:48:00 +02:00
2010-03-31 07:41:31 +00:00
// make sure the guest account and not-logged-in users never get any risky caps no matter what the actual settings are.
2011-10-14 12:48:00 +02:00
if (( $capinfo -> captype === 'write' ) or ( $capinfo -> riskbitmask & ( RISK_XSS | RISK_CONFIG | RISK_DATALOSS ))) {
2010-03-31 07:41:31 +00:00
if ( isguestuser ( $userid ) or $userid == 0 ) {
return false ;
2009-06-29 05:00:45 +00:00
}
2007-09-19 07:02:31 +00:00
}
2011-10-14 12:48:00 +02:00
// somehow make sure the user is not deleted and actually exists
if ( $userid != 0 ) {
if ( $userid == $USER -> id and isset ( $USER -> deleted )) {
// this prevents one query per page, it is a bit of cheating,
// but hopefully session is terminated properly once user is deleted
if ( $USER -> deleted ) {
return false ;
}
2007-10-09 12:49:54 +00:00
} else {
2011-10-14 12:48:00 +02:00
if ( ! context_user :: instance ( $userid , IGNORE_MISSING )) {
// no user context == invalid userid
return false ;
}
2007-11-16 03:34:05 +00:00
}
2007-09-19 07:10:24 +00:00
}
2007-10-09 12:49:54 +00:00
2011-10-14 12:48:00 +02:00
// context path/depth must be valid
if ( empty ( $context -> path ) or $context -> depth == 0 ) {
// this should not happen often, each upgrade tries to rebuild the context paths
debugging ( 'Context id ' . $context -> id . ' does not have valid path, please use build_context_path()' );
if ( is_siteadmin ( $userid )) {
return true ;
2007-10-09 12:49:54 +00:00
} else {
2011-10-14 12:48:00 +02:00
return false ;
2007-10-09 12:49:54 +00:00
}
2007-09-19 07:10:24 +00:00
}
2007-10-09 12:49:54 +00:00
2010-03-31 07:41:31 +00:00
// Find out if user is admin - it is not possible to override the doanything in any way
// and it is not possible to switch to admin role either.
if ( $doanything ) {
if ( is_siteadmin ( $userid )) {
2010-07-07 08:21:51 +00:00
if ( $userid != $USER -> id ) {
return true ;
}
// make sure switchrole is not used in this context
if ( empty ( $USER -> access [ 'rsw' ])) {
return true ;
}
$parts = explode ( '/' , trim ( $context -> path , '/' ));
$path = '' ;
$switched = false ;
foreach ( $parts as $part ) {
$path .= '/' . $part ;
if ( ! empty ( $USER -> access [ 'rsw' ][ $path ])) {
$switched = true ;
break ;
}
}
if ( ! $switched ) {
return true ;
}
//ok, admin switched role in this context, let's use normal access control rules
2010-03-31 07:41:31 +00:00
}
}
2011-10-14 12:48:00 +02:00
// Careful check for staleness...
$context -> reload_if_dirty ();
2007-09-19 07:03:34 +00:00
2011-10-14 12:48:00 +02:00
if ( $USER -> id == $userid ) {
if ( ! isset ( $USER -> access )) {
load_all_capabilities ();
2007-09-19 07:02:31 +00:00
}
2011-10-14 12:48:00 +02:00
$access =& $USER -> access ;
2007-09-20 04:52:54 +00:00
2011-10-14 12:48:00 +02:00
} else {
// make sure user accessdata is really loaded
get_user_accessdata ( $userid , true );
$access =& $ACCESSLIB_PRIVATE -> accessdatabyuser [ $userid ];
2007-09-19 07:06:55 +00:00
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
// Load accessdata for below-the-course context if necessary,
// all contexts at and above all courses are already loaded
if ( $context -> contextlevel != CONTEXT_COURSE and $coursecontext = $context -> get_course_context ( false )) {
load_course_context ( $userid , $coursecontext , $access );
2007-09-19 07:26:02 +00:00
}
2011-10-14 12:48:00 +02:00
return has_capability_in_accessdata ( $capability , $context , $access );
2007-09-19 07:02:31 +00:00
}
2007-10-30 10:50:20 +00:00
/**
2009-07-21 06:24:44 +00:00
* Check if the user has any one of several capabilities from a list .
2009-05-19 07:50:54 +00:00
*
2009-07-21 06:24:44 +00:00
* This is just a utility method that calls has_capability in a loop . Try to put
* the capabilities that most users are likely to have first in the list for best
* performance .
2007-10-30 10:50:20 +00:00
*
2012-01-14 19:12:43 +01:00
* @ category access
2009-05-19 07:50:54 +00:00
* @ see has_capability ()
2012-01-14 19:12:43 +01:00
*
2009-07-21 06:24:44 +00:00
* @ param array $capabilities an array of capability names .
2012-01-14 19:12:43 +01:00
* @ param context $context the context to check the capability in . You normally get this with instance method of a context class .
* @ param integer | stdClass $user A user id or object . By default ( null ) checks the permissions of the current user .
2010-03-31 07:41:31 +00:00
* @ param boolean $doanything If false , ignore effect of admin role assignment
2009-07-21 06:24:44 +00:00
* @ return boolean true if the user has any of these capabilities . Otherwise false .
2007-10-30 10:50:20 +00:00
*/
2012-01-14 19:12:43 +01:00
function has_any_capability ( array $capabilities , context $context , $user = null , $doanything = true ) {
2007-10-30 10:50:20 +00:00
foreach ( $capabilities as $capability ) {
2012-01-14 19:12:43 +01:00
if ( has_capability ( $capability , $context , $user , $doanything )) {
2007-10-30 10:50:20 +00:00
return true ;
}
}
return false ;
}
2008-12-08 07:28:19 +00:00
/**
2009-07-21 06:24:44 +00:00
* Check if the user has all the capabilities in a list .
2009-05-19 07:50:54 +00:00
*
2009-07-21 06:24:44 +00:00
* This is just a utility method that calls has_capability in a loop . Try to put
* the capabilities that fewest users are likely to have first in the list for best
* performance .
2008-12-08 07:28:19 +00:00
*
2012-01-14 19:12:43 +01:00
* @ category access
2009-05-19 07:50:54 +00:00
* @ see has_capability ()
2012-01-14 19:12:43 +01:00
*
2009-07-21 06:24:44 +00:00
* @ param array $capabilities an array of capability names .
2012-01-14 19:12:43 +01:00
* @ param context $context the context to check the capability in . You normally get this with instance method of a context class .
* @ param integer | stdClass $user A user id or object . By default ( null ) checks the permissions of the current user .
2010-03-31 07:41:31 +00:00
* @ param boolean $doanything If false , ignore effect of admin role assignment
2009-07-21 06:24:44 +00:00
* @ return boolean true if the user has all of these capabilities . Otherwise false .
2008-12-08 07:28:19 +00:00
*/
2012-01-14 19:12:43 +01:00
function has_all_capabilities ( array $capabilities , context $context , $user = null , $doanything = true ) {
2008-12-08 07:28:19 +00:00
foreach ( $capabilities as $capability ) {
2012-01-14 19:12:43 +01:00
if ( ! has_capability ( $capability , $context , $user , $doanything )) {
2008-12-08 07:28:19 +00:00
return false ;
}
}
return true ;
}
2007-10-09 12:49:54 +00:00
/**
2010-03-31 07:41:31 +00:00
* Check if the user is an admin at the site level .
2009-05-19 07:50:54 +00:00
*
2010-03-31 07:41:31 +00:00
* Please note that use of proper capabilities is always encouraged ,
* this function is supposed to be used from core or for temporary hacks .
2007-09-19 07:30:09 +00:00
*
2012-01-14 19:12:43 +01:00
* @ category access
*
2011-10-14 12:48:00 +02:00
* @ param int | stdClass $user_or_id user id or user object
* @ return bool true if user is one of the administrators , false otherwise
2007-09-19 07:30:09 +00:00
*/
2010-10-01 21:45:14 +00:00
function is_siteadmin ( $user_or_id = null ) {
2010-03-31 07:41:31 +00:00
global $CFG , $USER ;
2007-09-19 07:30:09 +00:00
2010-10-01 21:45:14 +00:00
if ( $user_or_id === null ) {
2010-03-31 07:41:31 +00:00
$user_or_id = $USER ;
}
2008-10-30 10:49:15 +00:00
2010-03-31 07:41:31 +00:00
if ( empty ( $user_or_id )) {
return false ;
}
if ( ! empty ( $user_or_id -> id )) {
$userid = $user_or_id -> id ;
} else {
$userid = $user_or_id ;
}
2008-10-30 10:49:15 +00:00
2010-03-31 07:41:31 +00:00
$siteadmins = explode ( ',' , $CFG -> siteadmins );
return in_array ( $userid , $siteadmins );
2008-10-30 10:49:15 +00:00
}
2008-11-18 07:10:00 +00:00
/**
2010-03-31 07:41:31 +00:00
* Returns true if user has at least one role assign
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
* of 'coursecontact' role ( is potentially listed in some course descriptions ) .
2010-10-01 21:45:14 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param int $userid
* @ return bool
2008-11-18 07:10:00 +00:00
*/
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
function has_coursecontact_role ( $userid ) {
2010-09-17 07:39:14 +00:00
global $DB , $CFG ;
2008-11-18 07:10:00 +00:00
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
if ( empty ( $CFG -> coursecontact )) {
2010-03-31 07:41:31 +00:00
return false ;
}
$sql = " SELECT 1
FROM { role_assignments }
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
WHERE userid = : userid AND roleid IN ( $CFG -> coursecontact ) " ;
2010-09-19 13:30:05 +00:00
return $DB -> record_exists_sql ( $sql , array ( 'userid' => $userid ));
2008-11-18 07:10:00 +00:00
}
2007-10-09 12:49:54 +00:00
/**
2010-03-07 09:28:54 +00:00
* Does the user have a capability to do something ?
2009-05-19 07:50:54 +00:00
*
2007-09-19 07:15:12 +00:00
* Walk the accessdata array and return true / false .
2011-10-14 12:48:00 +02:00
* Deals with prohibits , role switching , aggregating
2007-09-19 07:05:48 +00:00
* capabilities , etc .
*
* The main feature of here is being FAST and with no
2008-05-24 18:35:48 +00:00
* side effects .
2007-09-19 07:05:48 +00:00
*
2007-09-19 07:16:18 +00:00
* Notes :
*
2007-09-19 07:18:23 +00:00
* Switch Role merges with default role
* ------------------------------------
* If you are a teacher in course X , you have at least
* teacher - in - X + defaultloggedinuser - sitewide . So in the
* course you ' ll have techer + defaultloggedinuser .
* We try to mimic that in switchrole .
*
2010-03-07 09:28:54 +00:00
* Permission evaluation
* ---------------------
2010-05-22 18:46:43 +00:00
* Originally there was an extremely complicated way
2010-03-07 09:28:54 +00:00
* to determine the user access that dealt with
2010-05-22 18:46:43 +00:00
* " locality " or role assignments and role overrides .
* Now we simply evaluate access for each role separately
2010-03-07 09:28:54 +00:00
* and then verify if user has at least one role with allow
* and at the same time no role with prohibit .
*
2012-01-14 19:12:43 +01:00
* @ access private
2009-05-19 07:50:54 +00:00
* @ param string $capability
2011-10-14 12:48:00 +02:00
* @ param context $context
2009-05-19 07:50:54 +00:00
* @ param array $accessdata
* @ return bool
2007-09-19 07:05:48 +00:00
*/
2011-10-14 12:48:00 +02:00
function has_capability_in_accessdata ( $capability , context $context , array & $accessdata ) {
2007-09-19 07:16:18 +00:00
global $CFG ;
2010-03-07 09:28:54 +00:00
// Build $paths as a list of current + all parent "paths" with order bottom-to-top
2011-10-14 12:48:00 +02:00
$path = $context -> path ;
$paths = array ( $path );
while ( $path = rtrim ( $path , '0123456789' )) {
$path = rtrim ( $path , '/' );
if ( $path === '' ) {
break ;
}
$paths [] = $path ;
2007-09-19 07:16:18 +00:00
}
2010-03-07 09:28:54 +00:00
$roles = array ();
$switchedrole = false ;
2007-09-19 07:05:48 +00:00
2010-03-07 09:28:54 +00:00
// Find out if role switched
if ( ! empty ( $accessdata [ 'rsw' ])) {
2007-09-19 07:05:48 +00:00
// From the bottom up...
2010-03-07 09:28:54 +00:00
foreach ( $paths as $path ) {
2010-03-24 13:10:10 +00:00
if ( isset ( $accessdata [ 'rsw' ][ $path ])) {
2010-03-07 09:28:54 +00:00
// Found a switchrole assignment - check for that role _plus_ the default user role
2010-10-01 21:45:14 +00:00
$roles = array ( $accessdata [ 'rsw' ][ $path ] => null , $CFG -> defaultuserroleid => null );
2010-03-07 09:28:54 +00:00
$switchedrole = true ;
break ;
2007-09-19 07:05:48 +00:00
}
}
}
2010-03-07 09:28:54 +00:00
if ( ! $switchedrole ) {
// get all users roles in this context and above
foreach ( $paths as $path ) {
if ( isset ( $accessdata [ 'ra' ][ $path ])) {
foreach ( $accessdata [ 'ra' ][ $path ] as $roleid ) {
2010-10-01 21:45:14 +00:00
$roles [ $roleid ] = null ;
2007-09-19 07:02:31 +00:00
}
2010-03-07 09:28:54 +00:00
}
}
2007-09-19 07:02:31 +00:00
}
2010-03-07 09:28:54 +00:00
// Now find out what access is given to each role, going bottom-->up direction
2011-10-14 12:48:00 +02:00
$allowed = false ;
2010-03-07 09:28:54 +00:00
foreach ( $roles as $roleid => $ignored ) {
foreach ( $paths as $path ) {
if ( isset ( $accessdata [ 'rdef' ][ " { $path } : $roleid " ][ $capability ])) {
$perm = ( int ) $accessdata [ 'rdef' ][ " { $path } : $roleid " ][ $capability ];
2011-10-14 12:48:00 +02:00
if ( $perm === CAP_PROHIBIT ) {
// any CAP_PROHIBIT found means no permission for the user
return false ;
}
if ( is_null ( $roles [ $roleid ])) {
2010-03-07 09:28:54 +00:00
$roles [ $roleid ] = $perm ;
}
}
2007-09-19 07:02:31 +00:00
}
2011-10-14 12:48:00 +02:00
// CAP_ALLOW in any role means the user has a permission, we continue only to detect prohibits
$allowed = ( $allowed or $roles [ $roleid ] === CAP_ALLOW );
2007-09-19 07:02:31 +00:00
}
2011-10-14 12:48:00 +02:00
return $allowed ;
2007-09-19 07:03:19 +00:00
}
2006-08-14 05:55:40 +00:00
/**
2009-07-21 06:24:44 +00:00
* A convenience function that tests has_capability , and displays an error if
* the user does not have that capability .
2009-07-04 14:22:11 +00:00
*
2009-07-21 06:24:44 +00:00
* NOTE before Moodle 2.0 , this function attempted to make an appropriate
* require_login call before checking the capability . This is no longer the case .
* You must call require_login ( or one of its variants ) if you want to check the
* user is logged in , before you call this function .
2007-09-19 07:24:43 +00:00
*
2009-05-19 07:50:54 +00:00
* @ see has_capability ()
*
2009-07-21 06:24:44 +00:00
* @ param string $capability the name of the capability to check . For example mod / forum : view
2011-10-14 12:48:00 +02:00
* @ param context $context the context to check the capability in . You normally get this with { @ link get_context_instance } .
* @ param int $userid A user id . By default ( null ) checks the permissions of the current user .
2010-03-31 07:41:31 +00:00
* @ param bool $doanything If false , ignore effect of admin role assignment
2011-10-14 12:48:00 +02:00
* @ param string $errormessage The error string to to user . Defaults to 'nopermissions' .
2009-07-21 06:24:44 +00:00
* @ param string $stringfile The language file to load the error string from . Defaults to 'error' .
* @ return void terminates with an error if the user does not have the given capability .
2006-08-14 05:55:40 +00:00
*/
2011-10-14 12:48:00 +02:00
function require_capability ( $capability , context $context , $userid = null , $doanything = true ,
2009-07-21 06:24:44 +00:00
$errormessage = 'nopermissions' , $stringfile = '' ) {
2006-09-13 06:35:25 +00:00
if ( ! has_capability ( $capability , $context , $userid , $doanything )) {
2009-09-14 23:52:08 +00:00
throw new required_capability_exception ( $context , $capability , $errormessage , $stringfile );
2006-08-14 05:55:40 +00:00
}
}
2007-09-19 07:02:04 +00:00
/**
2009-05-19 07:50:54 +00:00
* Return a nested array showing role assignments
2007-09-19 07:02:04 +00:00
* all relevant role capabilities for the user at
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
* site / course_category / course levels
2007-09-19 07:02:04 +00:00
*
* We do _not_ delve deeper than courses because the number of
2011-10-14 12:48:00 +02:00
* overrides at the module / block levels can be HUGE .
2007-09-19 07:02:04 +00:00
*
2011-10-14 12:48:00 +02:00
* [ ra ] => [ / path ][ roleid ] = roleid
* [ rdef ] => [ / path : roleid ][ capability ] = permission
2007-09-19 07:02:04 +00:00
*
2012-01-14 19:12:43 +01:00
* @ access private
2010-10-01 21:45:14 +00:00
* @ param int $userid - the id of the user
2011-10-14 12:48:00 +02:00
* @ return array access info array
2007-09-19 07:02:04 +00:00
*/
2007-09-19 07:02:44 +00:00
function get_user_access_sitewide ( $userid ) {
2011-10-14 12:48:00 +02:00
global $CFG , $DB , $ACCESSLIB_PRIVATE ;
2007-09-19 07:02:04 +00:00
2011-10-14 12:48:00 +02:00
/* Get in a few cheap DB queries ...
2009-06-16 15:34:55 +00:00
* - role assignments
2007-09-19 07:02:04 +00:00
* - relevant role caps
2009-06-16 15:34:55 +00:00
* - above and within this user ' s RAs
2007-09-19 07:02:04 +00:00
* - below this user ' s RAs - limited to course level
*/
2011-10-14 12:48:00 +02:00
// raparents collects paths & roles we need to walk up the parenthood to build the minimal rdef
$raparents = array ();
$accessdata = get_empty_accessdata ();
2007-09-19 07:02:04 +00:00
2011-10-14 12:48:00 +02:00
// start with the default role
if ( ! empty ( $CFG -> defaultuserroleid )) {
$syscontext = context_system :: instance ();
$accessdata [ 'ra' ][ $syscontext -> path ][( int ) $CFG -> defaultuserroleid ] = ( int ) $CFG -> defaultuserroleid ;
2011-12-16 14:22:25 +01:00
$raparents [ $CFG -> defaultuserroleid ][ $syscontext -> id ] = $syscontext -> id ;
2011-10-14 12:48:00 +02:00
}
// load the "default frontpage role"
if ( ! empty ( $CFG -> defaultfrontpageroleid )) {
$frontpagecontext = context_course :: instance ( get_site () -> id );
if ( $frontpagecontext -> path ) {
$accessdata [ 'ra' ][ $frontpagecontext -> path ][( int ) $CFG -> defaultfrontpageroleid ] = ( int ) $CFG -> defaultfrontpageroleid ;
2011-12-16 14:22:25 +01:00
$raparents [ $CFG -> defaultfrontpageroleid ][ $frontpagecontext -> id ] = $frontpagecontext -> id ;
2011-10-14 12:48:00 +02:00
}
}
// preload every assigned role at and above course context
2011-12-16 14:22:25 +01:00
$sql = " SELECT ctx.path, ra.roleid, ra.contextid
2008-05-15 21:40:00 +00:00
FROM { role_assignments } ra
2011-12-16 14:22:25 +01:00
JOIN { context } ctx
ON ctx . id = ra . contextid
LEFT JOIN { block_instances } bi
ON ( ctx . contextlevel = " .CONTEXT_BLOCK. " AND bi . id = ctx . instanceid )
LEFT JOIN { context } bpctx
ON ( bpctx . id = bi . parentcontextid )
WHERE ra . userid = : userid
AND ( ctx . contextlevel <= " .CONTEXT_COURSE. " OR bpctx . contextlevel < " .CONTEXT_COURSE. " ) " ;
2011-10-14 12:48:00 +02:00
$params = array ( 'userid' => $userid );
$rs = $DB -> get_recordset_sql ( $sql , $params );
foreach ( $rs as $ra ) {
// RAs leafs are arrays to support multi-role assignments...
$accessdata [ 'ra' ][ $ra -> path ][( int ) $ra -> roleid ] = ( int ) $ra -> roleid ;
2011-12-16 14:22:25 +01:00
$raparents [ $ra -> roleid ][ $ra -> contextid ] = $ra -> contextid ;
2007-09-19 07:02:04 +00:00
}
2011-10-14 12:48:00 +02:00
$rs -> close ();
2007-09-19 07:02:04 +00:00
2011-10-14 12:48:00 +02:00
if ( empty ( $raparents )) {
return $accessdata ;
2007-09-19 07:02:04 +00:00
}
2009-06-16 15:34:55 +00:00
2011-10-14 12:48:00 +02:00
// now get overrides of interesting roles in all interesting child contexts
// hopefully we will not run out of SQL limits here,
2011-12-16 14:22:25 +01:00
// users would have to have very many roles at/above course context...
2011-10-14 12:48:00 +02:00
$sqls = array ();
$params = array ();
2009-06-16 15:34:55 +00:00
2011-10-14 12:48:00 +02:00
static $cp = 0 ;
2011-12-16 14:22:25 +01:00
foreach ( $raparents as $roleid => $ras ) {
2011-10-14 12:48:00 +02:00
$cp ++ ;
2011-12-16 14:22:25 +01:00
list ( $sqlcids , $cids ) = $DB -> get_in_or_equal ( $ras , SQL_PARAMS_NAMED , 'c' . $cp . '_' );
$params = array_merge ( $params , $cids );
2011-10-14 12:48:00 +02:00
$params [ 'r' . $cp ] = $roleid ;
$sqls [] = " (SELECT ctx.path, rc.roleid, rc.capability, rc.permission
FROM { role_capabilities } rc
JOIN { context } ctx
ON ( ctx . id = rc . contextid )
JOIN { context } pctx
2011-12-16 14:22:25 +01:00
ON ( pctx . id $sqlcids
2011-10-14 12:48:00 +02:00
AND ( ctx . id = pctx . id
OR ctx . path LIKE " . $DB->sql_concat ('pctx.path', " '/%' " ). "
OR pctx . path LIKE " . $DB->sql_concat ('ctx.path', " '/%' " ). " ))
2011-12-16 14:22:25 +01:00
LEFT JOIN { block_instances } bi
ON ( ctx . contextlevel = " .CONTEXT_BLOCK. " AND bi . id = ctx . instanceid )
LEFT JOIN { context } bpctx
ON ( bpctx . id = bi . parentcontextid )
2011-10-14 12:48:00 +02:00
WHERE rc . roleid = : r { $cp }
2011-12-16 14:22:25 +01:00
AND ( ctx . contextlevel <= " .CONTEXT_COURSE. " OR bpctx . contextlevel < " .CONTEXT_COURSE. " )
) " ;
2011-10-14 12:48:00 +02:00
}
// fixed capability order is necessary for rdef dedupe
$rs = $DB -> get_recordset_sql ( implode ( " \n UNION \n " , $sqls ) . " ORDER BY capability " , $params );
2007-09-19 07:02:04 +00:00
2011-10-14 12:48:00 +02:00
foreach ( $rs as $rd ) {
$k = $rd -> path . ':' . $rd -> roleid ;
$accessdata [ 'rdef' ][ $k ][ $rd -> capability ] = ( int ) $rd -> permission ;
2007-09-19 07:02:04 +00:00
}
2011-10-14 12:48:00 +02:00
$rs -> close ();
2007-09-19 07:02:04 +00:00
2011-10-14 12:48:00 +02:00
// share the role definitions
foreach ( $accessdata [ 'rdef' ] as $k => $unused ) {
if ( ! isset ( $ACCESSLIB_PRIVATE -> rolepermissions [ $k ])) {
$ACCESSLIB_PRIVATE -> rolepermissions [ $k ] = $accessdata [ 'rdef' ][ $k ];
2007-11-12 16:53:53 +00:00
}
2011-10-14 12:48:00 +02:00
$accessdata [ 'rdef_count' ] ++ ;
$accessdata [ 'rdef' ][ $k ] =& $ACCESSLIB_PRIVATE -> rolepermissions [ $k ];
2007-09-19 07:02:04 +00:00
}
2011-10-14 12:48:00 +02:00
2007-09-20 04:52:54 +00:00
return $accessdata ;
2007-09-19 07:02:04 +00:00
}
2007-09-19 07:02:44 +00:00
/**
2011-10-14 12:48:00 +02:00
* Add to the access ctrl array the data needed by a user for a given course .
2007-09-19 07:02:44 +00:00
*
2011-10-14 12:48:00 +02:00
* This function injects all course related access info into the accessdata array .
*
2012-01-14 19:12:43 +01:00
* @ access private
2011-10-14 12:48:00 +02:00
* @ param int $userid the id of the user
* @ param context_course $coursecontext course context
* @ param array $accessdata accessdata array ( modified )
* @ return void modifies $accessdata parameter
2007-09-19 07:02:44 +00:00
*/
2011-10-14 12:48:00 +02:00
function load_course_context ( $userid , context_course $coursecontext , & $accessdata ) {
global $DB , $CFG , $ACCESSLIB_PRIVATE ;
2007-09-19 07:03:19 +00:00
2011-10-14 12:48:00 +02:00
if ( empty ( $coursecontext -> path )) {
// weird, this should not happen
return ;
}
2007-09-19 07:02:44 +00:00
2011-10-14 12:48:00 +02:00
if ( isset ( $accessdata [ 'loaded' ][ $coursecontext -> instanceid ])) {
// already loaded, great!
return ;
}
2007-09-19 07:02:44 +00:00
2011-10-14 12:48:00 +02:00
$roles = array ();
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
if ( empty ( $userid )) {
if ( ! empty ( $CFG -> notloggedinroleid )) {
$roles [ $CFG -> notloggedinroleid ] = $CFG -> notloggedinroleid ;
}
2007-09-19 07:02:44 +00:00
2011-10-14 12:48:00 +02:00
} else if ( isguestuser ( $userid )) {
if ( $guestrole = get_guest_role ()) {
$roles [ $guestrole -> id ] = $guestrole -> id ;
}
2007-09-19 07:02:44 +00:00
2011-10-14 12:48:00 +02:00
} else {
// Interesting role assignments at, above and below the course context
list ( $parentsaself , $params ) = $DB -> get_in_or_equal ( $coursecontext -> get_parent_context_ids ( true ), SQL_PARAMS_NAMED , 'pc_' );
$params [ 'userid' ] = $userid ;
$params [ 'children' ] = $coursecontext -> path . " /% " ;
$sql = " SELECT ra.*, ctx.path
FROM { role_assignments } ra
JOIN { context } ctx ON ra . contextid = ctx . id
WHERE ra . userid = : userid AND ( ctx . id $parentsaself OR ctx . path LIKE : children ) " ;
$rs = $DB -> get_recordset_sql ( $sql , $params );
// add missing role definitions
2008-05-15 21:40:00 +00:00
foreach ( $rs as $ra ) {
2011-10-14 12:48:00 +02:00
$accessdata [ 'ra' ][ $ra -> path ][( int ) $ra -> roleid ] = ( int ) $ra -> roleid ;
$roles [ $ra -> roleid ] = $ra -> roleid ;
2007-09-19 07:02:44 +00:00
}
2008-05-15 21:40:00 +00:00
$rs -> close ();
2007-09-19 07:02:44 +00:00
2011-10-14 12:48:00 +02:00
// add the "default frontpage role" when on the frontpage
if ( ! empty ( $CFG -> defaultfrontpageroleid )) {
$frontpagecontext = context_course :: instance ( get_site () -> id );
if ( $frontpagecontext -> id == $coursecontext -> id ) {
$roles [ $CFG -> defaultfrontpageroleid ] = $CFG -> defaultfrontpageroleid ;
}
}
2007-09-19 07:53:49 +00:00
2011-10-14 12:48:00 +02:00
// do not forget the default role
if ( ! empty ( $CFG -> defaultuserroleid )) {
$roles [ $CFG -> defaultuserroleid ] = $CFG -> defaultuserroleid ;
}
2007-09-19 07:02:44 +00:00
}
2011-10-14 12:48:00 +02:00
if ( ! $roles ) {
// weird, default roles must be missing...
$accessdata [ 'loaded' ][ $coursecontext -> instanceid ] = 1 ;
return ;
2007-09-21 04:33:50 +00:00
}
2011-10-14 12:48:00 +02:00
// now get overrides of interesting roles in all interesting contexts (this course + children + parents)
$params = array ( 'c' => $coursecontext -> id );
list ( $parentsaself , $rparams ) = $DB -> get_in_or_equal ( $coursecontext -> get_parent_context_ids ( true ), SQL_PARAMS_NAMED , 'pc_' );
$params = array_merge ( $params , $rparams );
list ( $roleids , $rparams ) = $DB -> get_in_or_equal ( $roles , SQL_PARAMS_NAMED , 'r_' );
$params = array_merge ( $params , $rparams );
2007-09-19 07:53:49 +00:00
$sql = " SELECT ctx.path, rc.roleid, rc.capability, rc.permission
2011-10-14 12:48:00 +02:00
FROM { role_capabilities } rc
JOIN { context } ctx
ON ( ctx . id = rc . contextid )
JOIN { context } cctx
ON ( cctx . id = : c
AND ( ctx . id $parentsaself OR ctx . path LIKE " . $DB->sql_concat ('cctx.path', " '/%' " ). " ))
WHERE rc . roleid $roleids
ORDER BY rc . capability " ; // fixed capability order is necessary for rdef dedupe
$rs = $DB -> get_recordset_sql ( $sql , $params );
2007-09-19 07:53:49 +00:00
2007-10-17 09:19:39 +00:00
$newrdefs = array ();
2010-12-30 01:48:43 +01:00
foreach ( $rs as $rd ) {
2011-10-14 12:48:00 +02:00
$k = $rd -> path . ':' . $rd -> roleid ;
if ( isset ( $accessdata [ 'rdef' ][ $k ])) {
continue ;
2007-09-19 07:02:44 +00:00
}
2011-10-14 12:48:00 +02:00
$newrdefs [ $k ][ $rd -> capability ] = ( int ) $rd -> permission ;
2007-09-19 07:02:44 +00:00
}
2010-12-30 01:48:43 +01:00
$rs -> close ();
2007-09-19 07:02:44 +00:00
2011-10-14 12:48:00 +02:00
// share new role definitions
foreach ( $newrdefs as $k => $unused ) {
if ( ! isset ( $ACCESSLIB_PRIVATE -> rolepermissions [ $k ])) {
$ACCESSLIB_PRIVATE -> rolepermissions [ $k ] = $newrdefs [ $k ];
}
$accessdata [ 'rdef_count' ] ++ ;
$accessdata [ 'rdef' ][ $k ] =& $ACCESSLIB_PRIVATE -> rolepermissions [ $k ];
2007-10-17 09:19:39 +00:00
}
2007-09-19 07:02:44 +00:00
2011-10-14 12:48:00 +02:00
$accessdata [ 'loaded' ][ $coursecontext -> instanceid ] = 1 ;
// we want to deduplicate the USER->access from time to time, this looks like a good place,
// because we have to do it before the end of session
dedupe_user_access ();
2007-09-19 07:02:44 +00:00
}
2006-10-23 15:17:31 +00:00
2007-03-07 13:45:41 +00:00
/**
2009-05-19 07:50:54 +00:00
* Add to the access ctrl array the data needed by a role for a given context .
2007-09-19 07:05:32 +00:00
*
* The data is added in the rdef key .
* This role - centric function is useful for role_switching
2011-10-14 12:48:00 +02:00
* and temporary course roles .
2007-09-19 07:05:32 +00:00
*
2012-01-14 19:12:43 +01:00
* @ access private
2011-10-14 12:48:00 +02:00
* @ param int $roleid the id of the user
* @ param context $context needs path !
* @ param array $accessdata accessdata array ( is modified )
2009-05-19 07:50:54 +00:00
* @ return array
2007-09-19 07:05:32 +00:00
*/
2011-10-14 12:48:00 +02:00
function load_role_access_by_context ( $roleid , context $context , & $accessdata ) {
global $DB , $ACCESSLIB_PRIVATE ;
2007-09-19 07:05:32 +00:00
/* Get the relevant rolecaps into rdef
* - relevant role caps
* - at ctx and above
* - below this ctx
*/
2011-10-14 12:48:00 +02:00
if ( empty ( $context -> path )) {
// weird, this should not happen
return ;
2007-09-19 07:05:32 +00:00
}
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
list ( $parentsaself , $params ) = $DB -> get_in_or_equal ( $context -> get_parent_context_ids ( true ), SQL_PARAMS_NAMED , 'pc_' );
$params [ 'roleid' ] = $roleid ;
$params [ 'childpath' ] = $context -> path . '/%' ;
2007-09-19 07:05:32 +00:00
$sql = " SELECT ctx.path, rc.capability, rc.permission
2008-05-15 21:40:00 +00:00
FROM { role_capabilities } rc
2011-10-14 12:48:00 +02:00
JOIN { context } ctx ON ( rc . contextid = ctx . id )
WHERE rc . roleid = : roleid AND ( ctx . id $parentsaself OR ctx . path LIKE : childpath )
ORDER BY rc . capability " ; // fixed capability order is necessary for rdef dedupe
2010-12-30 01:48:43 +01:00
$rs = $DB -> get_recordset_sql ( $sql , $params );
2011-10-14 12:48:00 +02:00
$newrdefs = array ();
2010-12-30 01:48:43 +01:00
foreach ( $rs as $rd ) {
2011-10-14 12:48:00 +02:00
$k = $rd -> path . ':' . $roleid ;
if ( isset ( $accessdata [ 'rdef' ][ $k ])) {
continue ;
}
$newrdefs [ $k ][ $rd -> capability ] = ( int ) $rd -> permission ;
2007-09-19 07:05:32 +00:00
}
2010-12-30 01:48:43 +01:00
$rs -> close ();
2007-09-19 07:05:32 +00:00
2011-10-14 12:48:00 +02:00
// share new role definitions
foreach ( $newrdefs as $k => $unused ) {
if ( ! isset ( $ACCESSLIB_PRIVATE -> rolepermissions [ $k ])) {
$ACCESSLIB_PRIVATE -> rolepermissions [ $k ] = $newrdefs [ $k ];
}
$accessdata [ 'rdef_count' ] ++ ;
$accessdata [ 'rdef' ][ $k ] =& $ACCESSLIB_PRIVATE -> rolepermissions [ $k ];
}
}
/**
* Returns empty accessdata structure .
2011-10-25 14:20:22 +02:00
*
2012-01-14 19:12:43 +01:00
* @ access private
2011-10-14 12:48:00 +02:00
* @ return array empt accessdata
*/
function get_empty_accessdata () {
$accessdata = array (); // named list
$accessdata [ 'ra' ] = array ();
$accessdata [ 'rdef' ] = array ();
$accessdata [ 'rdef_count' ] = 0 ; // this bloody hack is necessary because count($array) is slooooowwww in PHP
$accessdata [ 'rdef_lcc' ] = 0 ; // rdef_count during the last compression
$accessdata [ 'loaded' ] = array (); // loaded course contexts
$accessdata [ 'time' ] = time ();
2012-01-15 00:14:44 +01:00
$accessdata [ 'rsw' ] = array ();
2011-10-14 12:48:00 +02:00
2007-09-20 04:52:54 +00:00
return $accessdata ;
2007-09-19 07:05:32 +00:00
}
2007-10-17 09:19:39 +00:00
/**
2011-10-14 12:48:00 +02:00
* Get accessdata for a given user .
2007-09-19 07:06:55 +00:00
*
2012-01-14 19:12:43 +01:00
* @ access private
2009-05-19 07:50:54 +00:00
* @ param int $userid
2011-10-14 12:48:00 +02:00
* @ param bool $preloadonly true means do not return access array
* @ return array accessdata
2008-05-24 18:35:48 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_user_accessdata ( $userid , $preloadonly = false ) {
global $CFG , $ACCESSLIB_PRIVATE , $USER ;
2007-09-19 07:06:55 +00:00
2011-10-14 12:48:00 +02:00
if ( ! empty ( $USER -> acces [ 'rdef' ]) and empty ( $ACCESSLIB_PRIVATE -> rolepermissions )) {
// share rdef from USER session with rolepermissions cache in order to conserve memory
foreach ( $USER -> acces [ 'rdef' ] as $k => $v ) {
$ACCESSLIB_PRIVATE -> rolepermissions [ $k ] =& $USER -> acces [ 'rdef' ][ $k ];
2007-10-05 15:06:38 +00:00
}
2011-10-14 12:48:00 +02:00
$ACCESSLIB_PRIVATE -> accessdatabyuser [ $USER -> id ] = $USER -> acces ;
2007-09-19 07:06:55 +00:00
}
2011-10-14 12:48:00 +02:00
if ( ! isset ( $ACCESSLIB_PRIVATE -> accessdatabyuser [ $userid ])) {
if ( empty ( $userid )) {
if ( ! empty ( $CFG -> notloggedinroleid )) {
$accessdata = get_role_access ( $CFG -> notloggedinroleid );
} else {
// weird
return get_empty_accessdata ();
}
} else if ( isguestuser ( $userid )) {
if ( $guestrole = get_guest_role ()) {
$accessdata = get_role_access ( $guestrole -> id );
} else {
//weird
return get_empty_accessdata ();
}
} else {
$accessdata = get_user_access_sitewide ( $userid ); // includes default role and frontpage role
2007-09-26 07:12:38 +00:00
}
2007-10-09 12:49:54 +00:00
2011-10-14 12:48:00 +02:00
$ACCESSLIB_PRIVATE -> accessdatabyuser [ $userid ] = $accessdata ;
}
2007-10-17 09:19:39 +00:00
2011-10-14 12:48:00 +02:00
if ( $preloadonly ) {
return ;
} else {
return $ACCESSLIB_PRIVATE -> accessdatabyuser [ $userid ];
}
2007-09-19 07:06:55 +00:00
}
2007-09-19 07:11:18 +00:00
2007-10-17 09:19:39 +00:00
/**
2011-10-14 12:48:00 +02:00
* Try to minimise the size of $USER -> access by eliminating duplicate override storage ,
* this function looks for contexts with the same overrides and shares them .
2009-05-22 02:03:55 +00:00
*
2012-01-14 19:12:43 +01:00
* @ access private
2011-10-14 12:48:00 +02:00
* @ return void
2007-10-17 09:19:39 +00:00
*/
2011-10-14 12:48:00 +02:00
function dedupe_user_access () {
global $USER ;
2007-10-17 09:19:39 +00:00
2011-10-14 12:48:00 +02:00
if ( CLI_SCRIPT ) {
// no session in CLI --> no compression necessary
return ;
}
2007-10-17 09:19:39 +00:00
2011-10-14 12:48:00 +02:00
if ( empty ( $USER -> access [ 'rdef_count' ])) {
// weird, this should not happen
return ;
}
// the rdef is growing only, we never remove stuff from it, the rdef_lcc helps us to detect new stuff in rdef
if ( $USER -> access [ 'rdef_count' ] - $USER -> access [ 'rdef_lcc' ] > 10 ) {
// do not compress after each change, wait till there is more stuff to be done
return ;
}
$hashmap = array ();
foreach ( $USER -> access [ 'rdef' ] as $k => $def ) {
$hash = sha1 ( serialize ( $def ));
if ( isset ( $hashmap [ $hash ])) {
$USER -> access [ 'rdef' ][ $k ] =& $hashmap [ $hash ];
} else {
$hashmap [ $hash ] =& $USER -> access [ 'rdef' ][ $k ];
2007-10-17 09:19:39 +00:00
}
}
2011-10-14 12:48:00 +02:00
$USER -> access [ 'rdef_lcc' ] = $USER -> access [ 'rdef_count' ];
2007-10-17 09:19:39 +00:00
}
2007-09-19 07:05:32 +00:00
/**
2009-05-19 07:50:54 +00:00
* A convenience function to completely load all the capabilities
2011-10-14 12:48:00 +02:00
* for the current user . It is called from has_capability () and functions change permissions .
*
* Call it only _after_ you ' ve setup $USER and called check_enrolment_plugins ();
2009-05-19 07:50:54 +00:00
* @ see check_enrolment_plugins ()
2009-11-01 11:31:16 +00:00
*
2012-01-14 19:12:43 +01:00
* @ access private
2010-10-01 21:45:14 +00:00
* @ return void
2006-10-23 15:17:31 +00:00
*/
function load_all_capabilities () {
2011-10-14 12:48:00 +02:00
global $USER ;
2006-08-08 05:13:06 +00:00
2009-01-15 22:09:57 +00:00
// roles not installed yet - we are in the middle of installation
2009-06-24 09:17:56 +00:00
if ( during_initial_install ()) {
2009-01-12 16:52:53 +00:00
return ;
}
2011-10-14 12:48:00 +02:00
if ( ! isset ( $USER -> id )) {
// this should not happen
$USER -> id = 0 ;
2006-10-23 15:17:31 +00:00
}
2007-09-19 07:17:08 +00:00
2011-10-14 12:48:00 +02:00
unset ( $USER -> access );
$USER -> access = get_user_accessdata ( $USER -> id );
// deduplicate the overrides to minimize session size
dedupe_user_access ();
2007-09-19 07:17:08 +00:00
// Clear to force a refresh
2011-10-14 12:48:00 +02:00
unset ( $USER -> mycourses );
2011-11-05 12:12:48 +01:00
// init/reset internal enrol caches - active course enrolments and temp access
$USER -> enrol = array ( 'enrolled' => array (), 'tempguest' => array ());
2006-08-08 05:13:06 +00:00
}
2007-09-19 07:11:18 +00:00
/**
2008-05-24 18:35:48 +00:00
* A convenience function to completely reload all the capabilities
2007-09-19 07:11:18 +00:00
* for the current user when roles have been updated in a relevant
2008-05-24 18:35:48 +00:00
* context -- but PRESERVING switchroles and loginas .
2011-10-14 12:48:00 +02:00
* This function resets all accesslib and context caches .
2007-09-19 07:11:18 +00:00
*
* That is - completely transparent to the user .
2008-05-24 18:35:48 +00:00
*
2011-10-14 12:48:00 +02:00
* Note : reloads $USER -> access completely .
2007-09-19 07:11:18 +00:00
*
2012-01-14 19:12:43 +01:00
* @ access private
2010-10-01 21:45:14 +00:00
* @ return void
2007-09-19 07:11:18 +00:00
*/
function reload_all_capabilities () {
2011-10-14 12:48:00 +02:00
global $USER , $DB , $ACCESSLIB_PRIVATE ;
2007-09-19 07:11:18 +00:00
// copy switchroles
$sw = array ();
2012-01-15 00:14:44 +01:00
if ( ! empty ( $USER -> access [ 'rsw' ])) {
2007-09-19 07:11:18 +00:00
$sw = $USER -> access [ 'rsw' ];
}
2011-10-14 12:48:00 +02:00
accesslib_clear_all_caches ( true );
2007-09-19 07:11:18 +00:00
unset ( $USER -> access );
2011-10-14 12:48:00 +02:00
$ACCESSLIB_PRIVATE -> dirtycontexts = array (); // prevent dirty flags refetching on this page
2008-05-24 18:35:48 +00:00
2007-09-19 07:11:18 +00:00
load_all_capabilities ();
foreach ( $sw as $path => $roleid ) {
2011-10-14 12:48:00 +02:00
if ( $record = $DB -> get_record ( 'context' , array ( 'path' => $path ))) {
$context = context :: instance_by_id ( $record -> id );
role_switch ( $roleid , $context );
}
2007-09-19 07:11:18 +00:00
}
}
2006-10-23 15:17:31 +00:00
2008-05-15 21:40:00 +00:00
/**
2011-10-14 12:48:00 +02:00
* Adds a temp role to current USER -> access array .
2007-09-19 07:16:29 +00:00
*
2011-10-14 12:48:00 +02:00
* Useful for the " temporary guest " access we grant to logged - in users .
2012-01-14 19:12:43 +01:00
* This is useful for enrol plugins only .
2007-09-19 07:16:29 +00:00
*
2012-01-14 19:12:43 +01:00
* @ since 2.2
2011-10-14 12:48:00 +02:00
* @ param context_course $coursecontext
2009-05-19 07:50:54 +00:00
* @ param int $roleid
2011-10-14 12:48:00 +02:00
* @ return void
2007-09-19 07:16:29 +00:00
*/
2011-10-14 12:48:00 +02:00
function load_temp_course_role ( context_course $coursecontext , $roleid ) {
2011-11-05 12:12:48 +01:00
global $USER , $SITE ;
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
if ( empty ( $roleid )) {
debugging ( 'invalid role specified in load_temp_course_role()' );
return ;
}
2007-09-19 07:16:29 +00:00
2011-11-05 12:12:48 +01:00
if ( $coursecontext -> instanceid == $SITE -> id ) {
debugging ( 'Can not use temp roles on the frontpage' );
return ;
}
2011-10-14 12:48:00 +02:00
if ( ! isset ( $USER -> access )) {
load_all_capabilities ();
2008-05-15 21:40:00 +00:00
}
2007-09-19 07:16:29 +00:00
2011-10-14 12:48:00 +02:00
$coursecontext -> reload_if_dirty ();
2007-09-19 07:16:29 +00:00
2011-10-14 12:48:00 +02:00
if ( isset ( $USER -> access [ 'ra' ][ $coursecontext -> path ][ $roleid ])) {
return ;
2007-09-19 07:16:29 +00:00
}
2011-10-14 12:48:00 +02:00
// load course stuff first
load_course_context ( $USER -> id , $coursecontext , $USER -> access );
$USER -> access [ 'ra' ][ $coursecontext -> path ][( int ) $roleid ] = ( int ) $roleid ;
load_role_access_by_context ( $roleid , $coursecontext , $USER -> access );
2007-09-19 07:16:29 +00:00
}
2007-03-05 11:27:01 +00:00
/**
2011-10-14 12:48:00 +02:00
* Removes any extra guest roles from current USER -> access array .
2012-01-14 19:12:43 +01:00
* This is useful for enrol plugins only .
2011-10-14 12:48:00 +02:00
*
2012-01-14 19:12:43 +01:00
* @ since 2.2
2011-10-14 12:48:00 +02:00
* @ param context_course $coursecontext
* @ return void
2006-09-16 13:54:57 +00:00
*/
2011-10-14 12:48:00 +02:00
function remove_temp_course_roles ( context_course $coursecontext ) {
2011-11-05 12:12:48 +01:00
global $DB , $USER , $SITE ;
if ( $coursecontext -> instanceid == $SITE -> id ) {
debugging ( 'Can not use temp roles on the frontpage' );
return ;
}
2011-10-14 12:48:00 +02:00
if ( empty ( $USER -> access [ 'ra' ][ $coursecontext -> path ])) {
//no roles here, weird
return ;
}
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
$sql = " SELECT DISTINCT ra.roleid AS id
FROM { role_assignments } ra
WHERE ra . contextid = : contextid AND ra . userid = : userid " ;
2011-10-14 12:48:00 +02:00
$ras = $DB -> get_records_sql ( $sql , array ( 'contextid' => $coursecontext -> id , 'userid' => $USER -> id ));
2006-09-25 01:34:21 +00:00
2011-10-14 12:48:00 +02:00
$USER -> access [ 'ra' ][ $coursecontext -> path ] = array ();
foreach ( $ras as $r ) {
$USER -> access [ 'ra' ][ $coursecontext -> path ][( int ) $r -> id ] = ( int ) $r -> id ;
2010-10-13 08:29:22 +00:00
}
2006-09-16 13:54:57 +00:00
}
2007-03-03 21:07:07 +00:00
/**
2010-03-31 07:41:31 +00:00
* Returns array of all role archetypes .
2009-05-22 02:03:55 +00:00
*
2009-05-19 07:50:54 +00:00
* @ return array
2007-03-03 21:07:07 +00:00
*/
2010-03-31 07:41:31 +00:00
function get_role_archetypes () {
2007-03-03 21:07:07 +00:00
return array (
2010-03-31 07:41:31 +00:00
'manager' => 'manager' ,
'coursecreator' => 'coursecreator' ,
'editingteacher' => 'editingteacher' ,
'teacher' => 'teacher' ,
'student' => 'student' ,
'guest' => 'guest' ,
'user' => 'user' ,
'frontpage' => 'frontpage'
2007-03-03 21:07:07 +00:00
);
}
2006-08-08 05:13:06 +00:00
/**
2010-05-22 18:46:43 +00:00
* Assign the defaults found in this capability definition to roles that have
2006-08-08 05:13:06 +00:00
* the corresponding legacy capabilities assigned to them .
2009-05-22 02:03:55 +00:00
*
2009-05-19 07:50:54 +00:00
* @ param string $capability
* @ param array $legacyperms an array in the format ( example ) :
2006-08-08 05:13:06 +00:00
* 'guest' => CAP_PREVENT ,
* 'student' => CAP_ALLOW ,
* 'teacher' => CAP_ALLOW ,
* 'editingteacher' => CAP_ALLOW ,
* 'coursecreator' => CAP_ALLOW ,
2010-03-31 07:41:31 +00:00
* 'manager' => CAP_ALLOW
2009-05-19 07:50:54 +00:00
* @ return boolean success or failure .
2006-08-08 05:13:06 +00:00
*/
function assign_legacy_capabilities ( $capability , $legacyperms ) {
2006-09-20 21:00:45 +00:00
2010-03-31 07:41:31 +00:00
$archetypes = get_role_archetypes ();
2007-03-03 21:07:07 +00:00
2006-08-08 05:13:06 +00:00
foreach ( $legacyperms as $type => $perm ) {
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
$systemcontext = context_system :: instance ();
2010-03-31 07:41:31 +00:00
if ( $type === 'admin' ) {
debugging ( 'Legacy type admin in access.php was renamed to manager, please update the code.' );
$type = 'manager' ;
}
2006-09-20 21:00:45 +00:00
2010-03-31 07:41:31 +00:00
if ( ! array_key_exists ( $type , $archetypes )) {
2008-05-15 03:07:21 +00:00
print_error ( 'invalidlegacy' , '' , '' , $type );
2007-03-03 21:07:07 +00:00
}
2006-09-20 21:00:45 +00:00
2010-03-31 07:41:31 +00:00
if ( $roles = get_archetype_roles ( $type )) {
2006-09-12 07:07:30 +00:00
foreach ( $roles as $role ) {
// Assign a site level capability.
if ( ! assign_capability ( $capability , $perm , $role -> id , $systemcontext -> id )) {
return false ;
}
2006-08-08 05:13:06 +00:00
}
}
}
return true ;
}
2008-11-05 08:17:30 +00:00
/**
2011-10-14 12:48:00 +02:00
* Verify capability risks .
*
2012-01-14 19:12:43 +01:00
* @ param stdClass $capability a capability - a row from the capabilities table .
2010-05-21 19:31:17 +00:00
* @ return boolean whether this capability is safe - that is , whether people with the
2008-11-05 08:17:30 +00:00
* safeoverrides capability should be allowed to change it .
*/
function is_safe_capability ( $capability ) {
2008-11-18 08:08:18 +00:00
return ! (( RISK_DATALOSS | RISK_MANAGETRUST | RISK_CONFIG | RISK_XSS | RISK_PERSONAL ) & $capability -> riskbitmask );
2008-11-05 08:17:30 +00:00
}
2006-08-14 07:15:03 +00:00
2006-08-08 05:13:06 +00:00
/**
2011-10-14 12:48:00 +02:00
* Get the local override ( if any ) for a given capability in a role in a context
2010-06-04 07:30:42 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param int $roleid
* @ param int $contextid
* @ param string $capability
* @ return stdClass local capability override
2006-08-08 05:13:06 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_local_override ( $roleid , $contextid , $capability ) {
global $DB ;
return $DB -> get_record ( 'role_capabilities' , array ( 'roleid' => $roleid , 'capability' => $capability , 'contextid' => $contextid ));
}
2007-09-19 07:13:08 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns context instance plus related course and cm instances
*
* @ param int $contextid
* @ return array of ( $context , $course , $cm )
*/
function get_context_info_array ( $contextid ) {
global $DB ;
2007-09-19 07:13:08 +00:00
2011-10-14 12:48:00 +02:00
$context = context :: instance_by_id ( $contextid , MUST_EXIST );
$course = null ;
$cm = null ;
2007-09-19 07:13:08 +00:00
2011-10-14 12:48:00 +02:00
if ( $context -> contextlevel == CONTEXT_COURSE ) {
$course = $DB -> get_record ( 'course' , array ( 'id' => $context -> instanceid ), '*' , MUST_EXIST );
2007-09-19 07:13:08 +00:00
2011-10-14 12:48:00 +02:00
} else if ( $context -> contextlevel == CONTEXT_MODULE ) {
$cm = get_coursemodule_from_id ( '' , $context -> instanceid , 0 , false , MUST_EXIST );
$course = $DB -> get_record ( 'course' , array ( 'id' => $cm -> course ), '*' , MUST_EXIST );
2007-10-05 15:06:38 +00:00
2011-10-14 12:48:00 +02:00
} else if ( $context -> contextlevel == CONTEXT_BLOCK ) {
$parent = $context -> get_parent_context ();
2008-09-06 13:17:15 +00:00
2011-10-14 12:48:00 +02:00
if ( $parent -> contextlevel == CONTEXT_COURSE ) {
$course = $DB -> get_record ( 'course' , array ( 'id' => $parent -> instanceid ), '*' , MUST_EXIST );
} else if ( $parent -> contextlevel == CONTEXT_MODULE ) {
$cm = get_coursemodule_from_id ( '' , $parent -> instanceid , 0 , false , MUST_EXIST );
$course = $DB -> get_record ( 'course' , array ( 'id' => $cm -> course ), '*' , MUST_EXIST );
}
2006-08-08 05:13:06 +00:00
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
return array ( $context , $course , $cm );
2006-08-08 05:13:06 +00:00
}
2007-03-05 11:27:01 +00:00
/**
2011-10-14 12:48:00 +02:00
* Function that creates a role
2009-05-19 07:50:54 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param string $name role name
* @ param string $shortname role short name
* @ param string $description role description
* @ param string $archetype
* @ return int id or dml_exception
2007-01-30 10:10:25 +00:00
*/
2011-10-14 12:48:00 +02:00
function create_role ( $name , $shortname , $description , $archetype = '' ) {
global $DB ;
2007-10-05 15:06:38 +00:00
2011-10-14 12:48:00 +02:00
if ( strpos ( $archetype , 'moodle/legacy:' ) !== false ) {
throw new coding_exception ( 'Use new role archetype parameter in create_role() instead of old legacy capabilities.' );
2007-01-30 10:10:25 +00:00
}
2007-10-05 15:06:38 +00:00
2011-10-14 12:48:00 +02:00
// verify role archetype actually exists
$archetypes = get_role_archetypes ();
if ( empty ( $archetypes [ $archetype ])) {
$archetype = '' ;
2007-10-05 15:06:38 +00:00
}
2011-10-14 12:48:00 +02:00
// Insert the role record.
$role = new stdClass ();
$role -> name = $name ;
$role -> shortname = $shortname ;
$role -> description = $description ;
$role -> archetype = $archetype ;
//find free sortorder number
$role -> sortorder = $DB -> get_field ( 'role' , 'MAX(sortorder) + 1' , array ());
if ( empty ( $role -> sortorder )) {
$role -> sortorder = 1 ;
2007-10-05 15:06:38 +00:00
}
2011-10-14 12:48:00 +02:00
$id = $DB -> insert_record ( 'role' , $role );
2007-10-05 15:06:38 +00:00
2011-10-14 12:48:00 +02:00
return $id ;
2007-01-30 10:10:25 +00:00
}
2007-10-09 16:07:15 +00:00
2006-09-23 06:10:48 +00:00
/**
2011-10-14 12:48:00 +02:00
* Function that deletes a role and cleanups up after it
2009-05-22 02:03:55 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param int $roleid id of role to delete
* @ return bool always true
2006-09-23 06:10:48 +00:00
*/
2011-10-14 12:48:00 +02:00
function delete_role ( $roleid ) {
global $DB ;
2007-10-09 16:07:15 +00:00
2011-10-14 12:48:00 +02:00
// first unssign all users
role_unassign_all ( array ( 'roleid' => $roleid ));
2010-09-20 03:25:35 +00:00
2011-10-14 12:48:00 +02:00
// cleanup all references to this role, ignore errors
$DB -> delete_records ( 'role_capabilities' , array ( 'roleid' => $roleid ));
$DB -> delete_records ( 'role_allow_assign' , array ( 'roleid' => $roleid ));
$DB -> delete_records ( 'role_allow_assign' , array ( 'allowassign' => $roleid ));
$DB -> delete_records ( 'role_allow_override' , array ( 'roleid' => $roleid ));
$DB -> delete_records ( 'role_allow_override' , array ( 'allowoverride' => $roleid ));
$DB -> delete_records ( 'role_names' , array ( 'roleid' => $roleid ));
$DB -> delete_records ( 'role_context_levels' , array ( 'roleid' => $roleid ));
2010-05-20 07:53:48 +00:00
2011-10-14 12:48:00 +02:00
// finally delete the role itself
// get this before the name is gone for logging
$rolename = $DB -> get_field ( 'role' , 'name' , array ( 'id' => $roleid ));
2010-09-19 13:09:48 +00:00
2011-10-14 12:48:00 +02:00
$DB -> delete_records ( 'role' , array ( 'id' => $roleid ));
2010-09-19 13:09:48 +00:00
2011-10-14 12:48:00 +02:00
add_to_log ( SITEID , 'role' , 'delete' , 'admin/roles/action=delete&roleid=' . $roleid , $rolename , '' );
2010-05-20 07:40:17 +00:00
return true ;
2006-09-23 06:10:48 +00:00
}
2007-10-05 20:16:51 +00:00
/**
2011-10-14 12:48:00 +02:00
* Function to write context specific overrides , or default capabilities .
2009-05-22 02:03:55 +00:00
*
2011-10-14 12:48:00 +02:00
* NOTE : use $context -> mark_dirty () after this
*
* @ param string $capability string name
* @ param int $permission CAP_ constants
* @ param int $roleid role id
* @ param int | context $contextid context id
* @ param bool $overwrite
* @ return bool always true or exception
2007-10-05 20:16:51 +00:00
*/
2011-10-14 12:48:00 +02:00
function assign_capability ( $capability , $permission , $roleid , $contextid , $overwrite = false ) {
global $USER , $DB ;
2007-10-05 20:16:51 +00:00
2011-10-14 12:48:00 +02:00
if ( $contextid instanceof context ) {
$context = $contextid ;
} else {
$context = context :: instance_by_id ( $contextid );
2007-10-05 20:16:51 +00:00
}
2011-10-14 12:48:00 +02:00
if ( empty ( $permission ) || $permission == CAP_INHERIT ) { // if permission is not set
unassign_capability ( $capability , $roleid , $context -> id );
return true ;
2007-10-05 20:16:51 +00:00
}
2011-10-14 12:48:00 +02:00
$existing = $DB -> get_record ( 'role_capabilities' , array ( 'contextid' => $context -> id , 'roleid' => $roleid , 'capability' => $capability ));
2007-10-05 20:16:51 +00:00
2011-10-14 12:48:00 +02:00
if ( $existing and ! $overwrite ) { // We want to keep whatever is there already
return true ;
2007-10-05 20:16:51 +00:00
}
2011-10-14 12:48:00 +02:00
$cap = new stdClass ();
$cap -> contextid = $context -> id ;
$cap -> roleid = $roleid ;
$cap -> capability = $capability ;
$cap -> permission = $permission ;
$cap -> timemodified = time ();
$cap -> modifierid = empty ( $USER -> id ) ? 0 : $USER -> id ;
2009-07-13 08:37:34 +00:00
2011-10-14 12:48:00 +02:00
if ( $existing ) {
$cap -> id = $existing -> id ;
$DB -> update_record ( 'role_capabilities' , $cap );
} else {
if ( $DB -> record_exists ( 'context' , array ( 'id' => $context -> id ))) {
$DB -> insert_record ( 'role_capabilities' , $cap );
}
2007-10-05 20:16:51 +00:00
}
2011-10-14 12:48:00 +02:00
return true ;
2007-10-05 20:16:51 +00:00
}
2007-09-19 06:50:53 +00:00
/**
2011-10-14 12:48:00 +02:00
* Unassign a capability from a role .
2007-09-19 06:50:53 +00:00
*
2011-10-14 12:48:00 +02:00
* NOTE : use $context -> mark_dirty () after this
*
* @ param string $capability the name of the capability
* @ param int $roleid the role id
* @ param int | context $contextid null means all contexts
* @ return boolean true or exception
2007-09-19 06:50:53 +00:00
*/
2011-10-14 12:48:00 +02:00
function unassign_capability ( $capability , $roleid , $contextid = null ) {
2008-05-24 18:35:48 +00:00
global $DB ;
2007-09-19 06:50:53 +00:00
2011-10-14 12:48:00 +02:00
if ( ! empty ( $contextid )) {
if ( $contextid instanceof context ) {
$context = $contextid ;
} else {
$context = context :: instance_by_id ( $contextid );
}
// delete from context rel, if this is the last override in this context
$DB -> delete_records ( 'role_capabilities' , array ( 'capability' => $capability , 'roleid' => $roleid , 'contextid' => $context -> id ));
} else {
$DB -> delete_records ( 'role_capabilities' , array ( 'capability' => $capability , 'roleid' => $roleid ));
2007-09-19 06:50:53 +00:00
}
return true ;
}
2008-12-18 17:52:29 +00:00
/**
2011-10-14 12:48:00 +02:00
* Get the roles that have a given capability assigned to it
2008-12-18 17:52:29 +00:00
*
2011-10-14 12:48:00 +02:00
* This function does not resolve the actual permission of the capability .
* It just checks for permissions and overrides .
* Use get_roles_with_cap_in_context () if resolution is required .
*
2012-02-21 11:48:33 +13:00
* @ param string $capability capability name ( string )
* @ param string $permission optional , the permission defined for this capability
2011-10-14 12:48:00 +02:00
* either CAP_ALLOW , CAP_PREVENT or CAP_PROHIBIT . Defaults to null which means any .
2012-02-21 11:48:33 +13:00
* @ param stdClass $context null means any
2011-10-14 12:48:00 +02:00
* @ return array of role records
2008-12-18 17:52:29 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_roles_with_capability ( $capability , $permission = null , $context = null ) {
global $DB ;
2008-12-18 17:52:29 +00:00
2011-10-14 12:48:00 +02:00
if ( $context ) {
$contexts = $context -> get_parent_context_ids ( true );
list ( $insql , $params ) = $DB -> get_in_or_equal ( $contexts , SQL_PARAMS_NAMED , 'ctx' );
$contextsql = " AND rc.contextid $insql " ;
} else {
$params = array ();
$contextsql = '' ;
2008-12-18 17:52:29 +00:00
}
2011-10-14 12:48:00 +02:00
if ( $permission ) {
$permissionsql = " AND rc.permission = :permission " ;
$params [ 'permission' ] = $permission ;
} else {
$permissionsql = '' ;
2008-12-18 17:52:29 +00:00
}
2011-10-14 12:48:00 +02:00
$sql = " SELECT r.*
FROM { role } r
WHERE r . id IN ( SELECT rc . roleid
FROM { role_capabilities } rc
WHERE rc . capability = : capname
$contextsql
$permissionsql ) " ;
$params [ 'capname' ] = $capability ;
return $DB -> get_records_sql ( $sql , $params );
2008-12-18 17:52:29 +00:00
}
2006-08-08 05:13:06 +00:00
/**
2011-10-14 12:48:00 +02:00
* This function makes a role - assignment ( a role for a user in a particular context )
2009-05-19 07:50:54 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param int $roleid the role of the id
* @ param int $userid userid
* @ param int | context $contextid id of the context
* @ param string $component example 'enrol_ldap' , defaults to '' which means manual assignment ,
* @ param int $itemid id of enrolment / auth plugin
* @ param string $timemodified defaults to current time
* @ return int new / existing id of the assignment
2006-08-08 05:13:06 +00:00
*/
2011-10-14 12:48:00 +02:00
function role_assign ( $roleid , $userid , $contextid , $component = '' , $itemid = 0 , $timemodified = '' ) {
global $USER , $DB ;
2006-08-23 01:29:52 +00:00
2011-10-14 12:48:00 +02:00
// first of all detect if somebody is using old style parameters
if ( $contextid === 0 or is_numeric ( $component )) {
throw new coding_exception ( 'Invalid call to role_assign(), code needs to be updated to use new order of parameters' );
2007-01-30 10:10:25 +00:00
}
2011-10-14 12:48:00 +02:00
// now validate all parameters
if ( empty ( $roleid )) {
throw new coding_exception ( 'Invalid call to role_assign(), roleid can not be empty' );
2006-09-27 17:58:17 +00:00
}
2011-10-14 12:48:00 +02:00
if ( empty ( $userid )) {
throw new coding_exception ( 'Invalid call to role_assign(), userid can not be empty' );
}
2010-10-26 16:21:23 +00:00
2011-10-14 12:48:00 +02:00
if ( $itemid ) {
if ( strpos ( $component , '_' ) === false ) {
throw new coding_exception ( 'Invalid call to role_assign(), component must start with plugin type such as"enrol_" when itemid specified' , 'component:' . $component );
2008-02-13 17:03:25 +00:00
}
2011-10-14 12:48:00 +02:00
} else {
$itemid = 0 ;
if ( $component !== '' and strpos ( $component , '_' ) === false ) {
throw new coding_exception ( 'Invalid call to role_assign(), invalid component string' , 'component:' . $component );
2008-02-13 17:03:25 +00:00
}
2011-10-14 12:48:00 +02:00
}
2008-02-13 17:03:25 +00:00
2011-10-14 12:48:00 +02:00
if ( ! $DB -> record_exists ( 'user' , array ( 'id' => $userid , 'deleted' => 0 ))) {
throw new coding_exception ( 'User ID does not exist or is deleted!' , 'userid:' . $userid );
}
2008-02-13 17:03:25 +00:00
2011-10-14 12:48:00 +02:00
if ( $contextid instanceof context ) {
$context = $contextid ;
} else {
$context = context :: instance_by_id ( $contextid , MUST_EXIST );
2006-08-13 13:28:01 +00:00
}
2011-10-14 12:48:00 +02:00
if ( ! $timemodified ) {
$timemodified = time ();
}
2008-02-13 17:03:25 +00:00
2011-12-20 22:36:31 +01:00
// Check for existing entry
2011-10-14 12:48:00 +02:00
$ras = $DB -> get_records ( 'role_assignments' , array ( 'roleid' => $roleid , 'contextid' => $context -> id , 'userid' => $userid , 'component' => $component , 'itemid' => $itemid ), 'id' );
2008-02-13 17:03:25 +00:00
2011-10-14 12:48:00 +02:00
if ( $ras ) {
// role already assigned - this should not happen
if ( count ( $ras ) > 1 ) {
// very weird - remove all duplicates!
$ra = array_shift ( $ras );
foreach ( $ras as $r ) {
$DB -> delete_records ( 'role_assignments' , array ( 'id' => $r -> id ));
}
} else {
$ra = reset ( $ras );
2008-02-13 17:03:25 +00:00
}
2006-08-13 13:28:01 +00:00
2011-10-14 12:48:00 +02:00
// actually there is no need to update, reset anything or trigger any event, so just return
return $ra -> id ;
}
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
// Create a new entry
$ra = new stdClass ();
$ra -> roleid = $roleid ;
$ra -> contextid = $context -> id ;
$ra -> userid = $userid ;
$ra -> component = $component ;
$ra -> itemid = $itemid ;
$ra -> timemodified = $timemodified ;
$ra -> modifierid = empty ( $USER -> id ) ? 0 : $USER -> id ;
2008-02-13 17:03:25 +00:00
2011-10-14 12:48:00 +02:00
$ra -> id = $DB -> insert_record ( 'role_assignments' , $ra );
2008-02-13 17:03:25 +00:00
2011-10-14 12:48:00 +02:00
// mark context as dirty - again expensive, but needed
$context -> mark_dirty ();
2008-02-13 17:03:25 +00:00
2011-10-14 12:48:00 +02:00
if ( ! empty ( $USER -> id ) && $USER -> id == $userid ) {
// If the user is the current user, then do full reload of capabilities too.
reload_all_capabilities ();
2006-09-08 19:28:03 +00:00
}
2006-08-14 05:55:40 +00:00
2011-10-14 12:48:00 +02:00
events_trigger ( 'role_assigned' , $ra );
2006-08-08 05:13:06 +00:00
2011-10-14 12:48:00 +02:00
return $ra -> id ;
}
2006-08-14 07:15:03 +00:00
2006-08-13 15:48:57 +00:00
/**
2011-10-14 12:48:00 +02:00
* Removes one role assignment
2009-05-22 02:03:55 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param int $roleid
* @ param int $userid
* @ param int | context $contextid
* @ param string $component
* @ param int $itemid
* @ return void
2006-08-13 15:48:57 +00:00
*/
2011-10-14 12:48:00 +02:00
function role_unassign ( $roleid , $userid , $contextid , $component = '' , $itemid = 0 ) {
// first make sure the params make sense
if ( $roleid == 0 or $userid == 0 or $contextid == 0 ) {
throw new coding_exception ( 'Invalid call to role_unassign(), please use role_unassign_all() when removing multiple role assignments' );
2006-08-13 15:48:57 +00:00
}
2011-10-14 12:48:00 +02:00
if ( $itemid ) {
if ( strpos ( $component , '_' ) === false ) {
throw new coding_exception ( 'Invalid call to role_assign(), component must start with plugin type such as "enrol_" when itemid specified' , 'component:' . $component );
}
} else {
$itemid = 0 ;
if ( $component !== '' and strpos ( $component , '_' ) === false ) {
throw new coding_exception ( 'Invalid call to role_assign(), invalid component string' , 'component:' . $component );
}
2006-08-13 15:48:57 +00:00
}
2011-10-14 12:48:00 +02:00
role_unassign_all ( array ( 'roleid' => $roleid , 'userid' => $userid , 'contextid' => $contextid , 'component' => $component , 'itemid' => $itemid ), false , false );
2006-08-13 15:48:57 +00:00
}
2006-08-13 09:25:45 +00:00
/**
2011-10-14 12:48:00 +02:00
* Removes multiple role assignments , parameters may contain :
* 'roleid' , 'userid' , 'contextid' , 'component' , 'enrolid' .
2009-05-22 02:03:55 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param array $params role assignment parameters
* @ param bool $subcontexts unassign in subcontexts too
* @ param bool $includemanual include manual role assignments too
* @ return void
2010-03-07 09:28:54 +00:00
*/
2011-10-14 12:48:00 +02:00
function role_unassign_all ( array $params , $subcontexts = false , $includemanual = false ) {
global $USER , $CFG , $DB ;
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
if ( ! $params ) {
throw new coding_exception ( 'Missing parameters in role_unsassign_all() call' );
}
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
$allowed = array ( 'roleid' , 'userid' , 'contextid' , 'component' , 'itemid' );
foreach ( $params as $key => $value ) {
if ( ! in_array ( $key , $allowed )) {
throw new coding_exception ( 'Unknown role_unsassign_all() parameter key' , 'key:' . $key );
2010-03-07 09:28:54 +00:00
}
}
2011-10-14 12:48:00 +02:00
if ( isset ( $params [ 'component' ]) and $params [ 'component' ] !== '' and strpos ( $params [ 'component' ], '_' ) === false ) {
throw new coding_exception ( 'Invalid component paramter in role_unsassign_all() call' , 'component:' . $params [ 'component' ]);
2011-02-13 18:59:53 +01:00
}
2011-10-14 12:48:00 +02:00
if ( $includemanual ) {
if ( ! isset ( $params [ 'component' ]) or $params [ 'component' ] === '' ) {
throw new coding_exception ( 'include manual parameter requires component parameter in role_unsassign_all() call' );
}
2010-07-30 20:51:01 +00:00
}
2011-10-14 12:48:00 +02:00
if ( $subcontexts ) {
if ( empty ( $params [ 'contextid' ])) {
throw new coding_exception ( 'subcontexts paramtere requires component parameter in role_unsassign_all() call' );
}
2010-07-30 20:51:01 +00:00
}
2011-10-14 12:48:00 +02:00
$ras = $DB -> get_records ( 'role_assignments' , $params );
foreach ( $ras as $ra ) {
$DB -> delete_records ( 'role_assignments' , array ( 'id' => $ra -> id ));
if ( $context = context :: instance_by_id ( $ra -> contextid , IGNORE_MISSING )) {
// this is a bit expensive but necessary
$context -> mark_dirty ();
2012-02-21 11:48:33 +13:00
// If the user is the current user, then do full reload of capabilities too.
2011-10-14 12:48:00 +02:00
if ( ! empty ( $USER -> id ) && $USER -> id == $ra -> userid ) {
reload_all_capabilities ();
}
}
events_trigger ( 'role_unassigned' , $ra );
2010-07-30 20:51:01 +00:00
}
2011-10-14 12:48:00 +02:00
unset ( $ras );
// process subcontexts
if ( $subcontexts and $context = context :: instance_by_id ( $params [ 'contextid' ], IGNORE_MISSING )) {
if ( $params [ 'contextid' ] instanceof context ) {
$context = $params [ 'contextid' ];
} else {
$context = context :: instance_by_id ( $params [ 'contextid' ], IGNORE_MISSING );
}
2010-07-30 20:51:01 +00:00
2011-10-14 12:48:00 +02:00
if ( $context ) {
$contexts = $context -> get_child_contexts ();
$mparams = $params ;
foreach ( $contexts as $context ) {
$mparams [ 'contextid' ] = $context -> id ;
$ras = $DB -> get_records ( 'role_assignments' , $mparams );
foreach ( $ras as $ra ) {
$DB -> delete_records ( 'role_assignments' , array ( 'id' => $ra -> id ));
// this is a bit expensive but necessary
$context -> mark_dirty ();
2012-02-21 11:48:33 +13:00
// If the user is the current user, then do full reload of capabilities too.
2011-10-14 12:48:00 +02:00
if ( ! empty ( $USER -> id ) && $USER -> id == $ra -> userid ) {
reload_all_capabilities ();
}
events_trigger ( 'role_unassigned' , $ra );
}
}
}
2010-07-30 20:51:01 +00:00
}
2011-10-14 12:48:00 +02:00
// do this once more for all manual role assignments
if ( $includemanual ) {
$params [ 'component' ] = '' ;
role_unassign_all ( $params , $subcontexts , false );
}
2010-07-30 20:51:01 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Determines if a user is currently logged in
*
2012-01-14 19:12:43 +01:00
* @ category access
*
2011-10-14 12:48:00 +02:00
* @ return bool
*/
function isloggedin () {
global $USER ;
2006-08-08 05:13:06 +00:00
2011-10-14 12:48:00 +02:00
return ( ! empty ( $USER -> id ));
}
2006-08-08 05:13:06 +00:00
2006-08-14 07:15:03 +00:00
/**
2011-10-14 12:48:00 +02:00
* Determines if a user is logged in as real guest user with username 'guest' .
2009-05-22 02:03:55 +00:00
*
2012-01-14 19:12:43 +01:00
* @ category access
*
2011-10-14 12:48:00 +02:00
* @ param int | object $user mixed user object or id , $USER if not specified
* @ return bool true if user is the real guest user , false if not logged in or other user
2006-08-08 05:13:06 +00:00
*/
2011-10-14 12:48:00 +02:00
function isguestuser ( $user = null ) {
global $USER , $DB , $CFG ;
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
// make sure we have the user id cached in config table, because we are going to use it a lot
if ( empty ( $CFG -> siteguest )) {
if ( ! $guestid = $DB -> get_field ( 'user' , 'id' , array ( 'username' => 'guest' , 'mnethostid' => $CFG -> mnet_localhost_id ))) {
// guest does not exist yet, weird
return false ;
}
set_config ( 'siteguest' , $guestid );
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
if ( $user === null ) {
$user = $USER ;
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
if ( $user === null ) {
// happens when setting the $USER
return false ;
2006-09-18 17:40:22 +00:00
2011-10-14 12:48:00 +02:00
} else if ( is_numeric ( $user )) {
return ( $CFG -> siteguest == $user );
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
} else if ( is_object ( $user )) {
if ( empty ( $user -> id )) {
return false ; // not logged in means is not be guest
} else {
return ( $CFG -> siteguest == $user -> id );
}
2006-09-21 22:34:45 +00:00
2011-10-14 12:48:00 +02:00
} else {
throw new coding_exception ( 'Invalid user parameter supplied for isguestuser() function!' );
}
2006-08-08 05:13:06 +00:00
}
2006-09-23 10:43:24 +00:00
/**
2011-10-14 12:48:00 +02:00
* Does user have a ( temporary or real ) guest access to course ?
2009-05-22 02:03:55 +00:00
*
2012-01-14 19:12:43 +01:00
* @ category access
*
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ param stdClass | int $user
* @ return bool
2006-09-23 10:43:24 +00:00
*/
2011-10-14 12:48:00 +02:00
function is_guest ( context $context , $user = null ) {
global $USER ;
2007-09-13 13:44:35 +00:00
2011-10-14 12:48:00 +02:00
// first find the course context
$coursecontext = $context -> get_course_context ();
2007-09-13 13:44:35 +00:00
2011-10-14 12:48:00 +02:00
// make sure there is a real user specified
if ( $user === null ) {
$userid = isset ( $USER -> id ) ? $USER -> id : 0 ;
} else {
$userid = is_object ( $user ) ? $user -> id : $user ;
}
2007-07-19 05:19:05 +00:00
2011-10-14 12:48:00 +02:00
if ( isguestuser ( $userid )) {
// can not inspect or be enrolled
return true ;
}
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
if ( has_capability ( 'moodle/course:view' , $coursecontext , $user )) {
// viewing users appear out of nowhere, they are neither guests nor participants
return false ;
}
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
// consider only real active enrolments here
if ( is_enrolled ( $coursecontext , $user , '' , true )) {
return false ;
}
2006-09-23 10:43:24 +00:00
2010-03-31 07:41:31 +00:00
return true ;
2006-09-23 10:43:24 +00:00
}
2006-08-08 05:13:06 +00:00
/**
2011-10-14 12:48:00 +02:00
* Returns true if the user has moodle / course : view capability in the course ,
* this is intended for admins , managers ( aka small admins ), inspectors , etc .
2009-05-19 07:50:54 +00:00
*
2012-01-14 19:12:43 +01:00
* @ category access
*
2011-10-14 12:48:00 +02:00
* @ param context $context
2012-02-21 11:48:33 +13:00
* @ param int | stdClass $user if null $USER is used
2011-10-14 12:48:00 +02:00
* @ param string $withcapability extra capability name
* @ return bool
2006-08-08 05:13:06 +00:00
*/
2011-10-14 12:48:00 +02:00
function is_viewing ( context $context , $user = null , $withcapability = '' ) {
// first find the course context
$coursecontext = $context -> get_course_context ();
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
if ( isguestuser ( $user )) {
// can not inspect
return false ;
2006-08-09 13:14:15 +00:00
}
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
if ( ! has_capability ( 'moodle/course:view' , $coursecontext , $user )) {
// admins are allowed to inspect courses
return false ;
2006-09-12 06:15:33 +00:00
}
2011-10-14 12:48:00 +02:00
if ( $withcapability and ! has_capability ( $withcapability , $context , $user )) {
// site admins always have the capability, but the enrolment above blocks
return false ;
2006-09-12 06:15:33 +00:00
}
2011-10-14 12:48:00 +02:00
2010-03-31 07:41:31 +00:00
return true ;
2006-08-08 05:13:06 +00:00
}
/**
2011-10-14 12:48:00 +02:00
* Returns true if user is enrolled ( is participating ) in course
* this is intended for students and teachers .
2009-11-01 11:31:16 +00:00
*
2011-11-05 12:12:48 +01:00
* Since 2.2 the result for active enrolments and current user are cached .
*
2012-01-14 19:12:43 +01:00
* @ package core_enrol
* @ category access
*
2011-10-14 12:48:00 +02:00
* @ param context $context
2012-02-21 11:48:33 +13:00
* @ param int | stdClass $user if null $USER is used , otherwise user object or id expected
2011-10-14 12:48:00 +02:00
* @ param string $withcapability extra capability name
* @ param bool $onlyactive consider only active enrolments in enabled plugins and time restrictions
* @ return bool
2006-08-08 05:13:06 +00:00
*/
2011-10-14 12:48:00 +02:00
function is_enrolled ( context $context , $user = null , $withcapability = '' , $onlyactive = false ) {
global $USER , $DB ;
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
// first find the course context
$coursecontext = $context -> get_course_context ();
// make sure there is a real user specified
if ( $user === null ) {
$userid = isset ( $USER -> id ) ? $USER -> id : 0 ;
2006-08-09 13:14:15 +00:00
} else {
2011-10-14 12:48:00 +02:00
$userid = is_object ( $user ) ? $user -> id : $user ;
2006-08-09 13:14:15 +00:00
}
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
if ( empty ( $userid )) {
// not-logged-in!
return false ;
} else if ( isguestuser ( $userid )) {
// guest account can not be enrolled anywhere
return false ;
2010-10-01 21:13:27 +00:00
}
2011-10-14 12:48:00 +02:00
if ( $coursecontext -> instanceid == SITEID ) {
// everybody participates on frontpage
2006-09-07 08:57:56 +00:00
} else {
2011-11-05 12:12:48 +01:00
// try cached info first - the enrolled flag is set only when active enrolment present
if ( $USER -> id == $userid ) {
$coursecontext -> reload_if_dirty ();
if ( isset ( $USER -> enrol [ 'enrolled' ][ $coursecontext -> instanceid ])) {
if ( $USER -> enrol [ 'enrolled' ][ $coursecontext -> instanceid ] > time ()) {
2012-06-20 17:02:00 +01:00
if ( $withcapability and ! has_capability ( $withcapability , $context , $userid )) {
return false ;
}
2011-11-05 12:12:48 +01:00
return true ;
2011-10-14 12:48:00 +02:00
}
}
2011-11-05 12:12:48 +01:00
}
if ( $onlyactive ) {
// look for active enrolments only
$until = enrol_get_enrolment_end ( $coursecontext -> instanceid , $userid );
if ( $until === false ) {
2011-10-14 12:48:00 +02:00
return false ;
}
2006-09-20 21:00:45 +00:00
2011-11-05 12:12:48 +01:00
if ( $USER -> id == $userid ) {
if ( $until == 0 ) {
$until = ENROL_MAX_TIMESTAMP ;
}
$USER -> enrol [ 'enrolled' ][ $coursecontext -> instanceid ] = $until ;
if ( isset ( $USER -> enrol [ 'tempguest' ][ $coursecontext -> instanceid ])) {
unset ( $USER -> enrol [ 'tempguest' ][ $coursecontext -> instanceid ]);
remove_temp_course_roles ( $coursecontext );
}
}
2011-10-14 12:48:00 +02:00
} else {
// any enrolment is good for us here, even outdated, disabled or inactive
$sql = " SELECT 'x'
FROM { user_enrolments } ue
JOIN { enrol } e ON ( e . id = ue . enrolid AND e . courseid = : courseid )
JOIN { user } u ON u . id = ue . userid
WHERE ue . userid = : userid AND u . deleted = 0 " ;
$params = array ( 'userid' => $userid , 'courseid' => $coursecontext -> instanceid );
if ( ! $DB -> record_exists_sql ( $sql , $params )) {
return false ;
}
}
}
2006-08-08 05:13:06 +00:00
2011-10-14 12:48:00 +02:00
if ( $withcapability and ! has_capability ( $withcapability , $context , $userid )) {
return false ;
}
2008-05-15 21:40:00 +00:00
2011-10-14 12:48:00 +02:00
return true ;
2006-08-08 05:13:06 +00:00
}
/**
2011-10-14 12:48:00 +02:00
* Returns true if the user is able to access the course .
2009-11-01 11:31:16 +00:00
*
2011-10-14 12:48:00 +02:00
* This function is in no way , shape , or form a substitute for require_login .
* It should only be used in circumstances where it is not possible to call require_login
* such as the navigation .
*
* This function checks many of the methods of access to a course such as the view
* capability , enrollments , and guest access . It also makes use of the cache
* generated by require_login for guest access .
*
* The flags within the $USER object that are used here should NEVER be used outside
* of this function can_access_course and require_login . Doing so WILL break future
* versions .
*
2011-10-29 09:45:54 +02:00
* @ param stdClass $course record
* @ param stdClass | int | null $user user record or id , current user if null
2011-10-14 12:48:00 +02:00
* @ param string $withcapability Check for this capability as well .
* @ param bool $onlyactive consider only active enrolments in enabled plugins and time restrictions
* @ return boolean Returns true if the user is able to access the course
*/
2011-10-29 09:45:54 +02:00
function can_access_course ( stdClass $course , $user = null , $withcapability = '' , $onlyactive = false ) {
2011-10-14 12:48:00 +02:00
global $DB , $USER ;
2006-09-20 21:00:45 +00:00
2011-10-29 09:45:54 +02:00
// this function originally accepted $coursecontext parameter
if ( $course instanceof context ) {
if ( $course instanceof context_course ) {
debugging ( 'deprecated context parameter, please use $course record' );
$coursecontext = $course ;
$course = $DB -> get_record ( 'course' , array ( 'id' => $coursecontext -> instanceid ));
} else {
debugging ( 'Invalid context parameter, please use $course record' );
return false ;
}
} else {
$coursecontext = context_course :: instance ( $course -> id );
}
if ( ! isset ( $USER -> id )) {
// should never happen
$USER -> id = 0 ;
2012-10-26 10:43:39 +08:00
debugging ( 'Course access check being performed on a user with no ID.' , DEBUG_DEVELOPER );
2011-10-29 09:45:54 +02:00
}
// make sure there is a user specified
if ( $user === null ) {
$userid = $USER -> id ;
} else {
$userid = is_object ( $user ) ? $user -> id : $user ;
}
unset ( $user );
2006-08-08 05:13:06 +00:00
2011-10-29 09:45:54 +02:00
if ( $withcapability and ! has_capability ( $withcapability , $coursecontext , $userid )) {
return false ;
}
if ( $userid == $USER -> id ) {
if ( ! empty ( $USER -> access [ 'rsw' ][ $coursecontext -> path ])) {
// the fact that somebody switched role means they can access the course no matter to what role they switched
return true ;
}
}
2011-10-30 10:53:42 +01:00
if ( ! $course -> visible and ! has_capability ( 'moodle/course:viewhiddencourses' , $coursecontext , $userid )) {
2011-10-29 09:45:54 +02:00
return false ;
}
if ( is_viewing ( $coursecontext , $userid )) {
2011-10-14 12:48:00 +02:00
return true ;
2006-08-08 05:13:06 +00:00
}
2011-10-29 09:45:54 +02:00
if ( $userid != $USER -> id ) {
// for performance reasons we do not verify temporary guest access for other users, sorry...
return is_enrolled ( $coursecontext , $userid , '' , $onlyactive );
}
2011-10-30 10:53:42 +01:00
// === from here we deal only with $USER ===
2011-11-05 12:12:48 +01:00
$coursecontext -> reload_if_dirty ();
2011-10-29 09:45:54 +02:00
if ( isset ( $USER -> enrol [ 'enrolled' ][ $course -> id ])) {
2011-11-05 12:12:48 +01:00
if ( $USER -> enrol [ 'enrolled' ][ $course -> id ] > time ()) {
2011-10-29 09:45:54 +02:00
return true ;
}
}
if ( isset ( $USER -> enrol [ 'tempguest' ][ $course -> id ])) {
2011-11-05 12:12:48 +01:00
if ( $USER -> enrol [ 'tempguest' ][ $course -> id ] > time ()) {
2011-10-29 09:45:54 +02:00
return true ;
}
}
2006-09-03 07:56:40 +00:00
2011-11-05 12:12:48 +01:00
if ( is_enrolled ( $coursecontext , $USER , '' , $onlyactive )) {
2011-10-29 09:45:54 +02:00
return true ;
2008-07-06 19:30:12 +00:00
}
2007-09-13 13:44:35 +00:00
2011-10-29 09:45:54 +02:00
// if not enrolled try to gain temporary guest access
$instances = $DB -> get_records ( 'enrol' , array ( 'courseid' => $course -> id , 'status' => ENROL_INSTANCE_ENABLED ), 'sortorder, id ASC' );
$enrols = enrol_get_plugins ( true );
foreach ( $instances as $instance ) {
if ( ! isset ( $enrols [ $instance -> enrol ])) {
continue ;
}
2011-11-05 12:12:48 +01:00
// Get a duration for the guest access, a timestamp in the future, 0 (always) or false.
2011-10-29 09:45:54 +02:00
$until = $enrols [ $instance -> enrol ] -> try_guestaccess ( $instance );
2011-11-05 12:12:48 +01:00
if ( $until !== false and $until > time ()) {
2011-10-29 09:45:54 +02:00
$USER -> enrol [ 'tempguest' ][ $course -> id ] = $until ;
return true ;
2011-10-14 12:48:00 +02:00
}
2006-09-17 08:42:42 +00:00
}
2011-11-05 12:12:48 +01:00
if ( isset ( $USER -> enrol [ 'tempguest' ][ $course -> id ])) {
unset ( $USER -> enrol [ 'tempguest' ][ $course -> id ]);
remove_temp_course_roles ( $coursecontext );
}
2011-10-29 09:45:54 +02:00
return false ;
2006-08-08 05:13:06 +00:00
}
/**
2011-10-14 12:48:00 +02:00
* Returns array with sql code and parameters returning all ids
* of users enrolled into course .
2009-05-22 02:03:55 +00:00
*
2011-10-14 12:48:00 +02:00
* This function is using 'eu[0-9]+_' prefix for table names and parameters .
*
2012-01-14 19:12:43 +01:00
* @ package core_enrol
* @ category access
*
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ param string $withcapability
* @ param int $groupid 0 means ignore groups , any other value limits the result by group id
* @ param bool $onlyactive consider only active enrolments in enabled plugins and time restrictions
* @ return array list ( $sql , $params )
2006-08-08 05:13:06 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_enrolled_sql ( context $context , $withcapability = '' , $groupid = 0 , $onlyactive = false ) {
global $DB , $CFG ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
// use unique prefix just in case somebody makes some SQL magic with the result
static $i = 0 ;
$i ++ ;
$prefix = 'eu' . $i . '_' ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
// first find the course context
$coursecontext = $context -> get_course_context ();
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$isfrontpage = ( $coursecontext -> instanceid == SITEID );
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$joins = array ();
$wheres = array ();
$params = array ();
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
list ( $contextids , $contextpaths ) = get_context_info_list ( $context );
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
// get all relevant capability info for all roles
if ( $withcapability ) {
list ( $incontexts , $cparams ) = $DB -> get_in_or_equal ( $contextids , SQL_PARAMS_NAMED , 'ctx' );
$cparams [ 'cap' ] = $withcapability ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$defs = array ();
$sql = " SELECT rc.id, rc.roleid, rc.permission, ctx.path
FROM { role_capabilities } rc
JOIN { context } ctx on rc . contextid = ctx . id
WHERE rc . contextid $incontexts AND rc . capability = : cap " ;
$rcs = $DB -> get_records_sql ( $sql , $cparams );
foreach ( $rcs as $rc ) {
$defs [ $rc -> path ][ $rc -> roleid ] = $rc -> permission ;
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$access = array ();
if ( ! empty ( $defs )) {
foreach ( $contextpaths as $path ) {
if ( empty ( $defs [ $path ])) {
continue ;
}
foreach ( $defs [ $path ] as $roleid => $perm ) {
if ( $perm == CAP_PROHIBIT ) {
$access [ $roleid ] = CAP_PROHIBIT ;
continue ;
}
if ( ! isset ( $access [ $roleid ])) {
$access [ $roleid ] = ( int ) $perm ;
}
}
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
}
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
unset ( $defs );
// make lists of roles that are needed and prohibited
$needed = array (); // one of these is enough
$prohibited = array (); // must not have any of these
foreach ( $access as $roleid => $perm ) {
if ( $perm == CAP_PROHIBIT ) {
unset ( $needed [ $roleid ]);
$prohibited [ $roleid ] = true ;
} else if ( $perm == CAP_ALLOW and empty ( $prohibited [ $roleid ])) {
$needed [ $roleid ] = true ;
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
}
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$defaultuserroleid = isset ( $CFG -> defaultuserroleid ) ? $CFG -> defaultuserroleid : 0 ;
$defaultfrontpageroleid = isset ( $CFG -> defaultfrontpageroleid ) ? $CFG -> defaultfrontpageroleid : 0 ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$nobody = false ;
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
2011-10-14 12:48:00 +02:00
if ( $isfrontpage ) {
if ( ! empty ( $prohibited [ $defaultuserroleid ]) or ! empty ( $prohibited [ $defaultfrontpageroleid ])) {
$nobody = true ;
} else if ( ! empty ( $needed [ $defaultuserroleid ]) or ! empty ( $needed [ $defaultfrontpageroleid ])) {
// everybody not having prohibit has the capability
$needed = array ();
} else if ( empty ( $needed )) {
$nobody = true ;
}
} else {
if ( ! empty ( $prohibited [ $defaultuserroleid ])) {
$nobody = true ;
} else if ( ! empty ( $needed [ $defaultuserroleid ])) {
// everybody not having prohibit has the capability
$needed = array ();
} else if ( empty ( $needed )) {
$nobody = true ;
}
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
if ( $nobody ) {
// nobody can match so return some SQL that does not return any results
$wheres [] = " 1 = 2 " ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
} else {
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
if ( $needed ) {
$ctxids = implode ( ',' , $contextids );
$roleids = implode ( ',' , array_keys ( $needed ));
$joins [] = " JOIN { role_assignments} { $prefix } ra3 ON ( { $prefix } ra3.userid = { $prefix } u.id AND { $prefix } ra3.roleid IN ( $roleids ) AND { $prefix } ra3.contextid IN ( $ctxids )) " ;
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
if ( $prohibited ) {
$ctxids = implode ( ',' , $contextids );
$roleids = implode ( ',' , array_keys ( $prohibited ));
$joins [] = " LEFT JOIN { role_assignments} { $prefix } ra4 ON ( { $prefix } ra4.userid = { $prefix } u.id AND { $prefix } ra4.roleid IN ( $roleids ) AND { $prefix } ra4.contextid IN ( $ctxids )) " ;
$wheres [] = " { $prefix } ra4.id IS NULL " ;
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
if ( $groupid ) {
$joins [] = " JOIN { groups_members} { $prefix } gm ON ( { $prefix } gm.userid = { $prefix } u.id AND { $prefix } gm.groupid = : { $prefix } gmid) " ;
$params [ " { $prefix } gmid " ] = $groupid ;
}
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
} else {
if ( $groupid ) {
$joins [] = " JOIN { groups_members} { $prefix } gm ON ( { $prefix } gm.userid = { $prefix } u.id AND { $prefix } gm.groupid = : { $prefix } gmid) " ;
$params [ " { $prefix } gmid " ] = $groupid ;
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$wheres [] = " { $prefix } u.deleted = 0 AND { $prefix } u.id <> : { $prefix } guestid " ;
$params [ " { $prefix } guestid " ] = $CFG -> siteguest ;
if ( $isfrontpage ) {
// all users are "enrolled" on the frontpage
2010-03-31 07:41:31 +00:00
} else {
2011-10-14 12:48:00 +02:00
$joins [] = " JOIN { user_enrolments} { $prefix } ue ON { $prefix } ue.userid = { $prefix } u.id " ;
$joins [] = " JOIN { enrol} { $prefix } e ON ( { $prefix } e.id = { $prefix } ue.enrolid AND { $prefix } e.courseid = : { $prefix } courseid) " ;
$params [ $prefix . 'courseid' ] = $coursecontext -> instanceid ;
if ( $onlyactive ) {
$wheres [] = " { $prefix } ue.status = : { $prefix } active AND { $prefix } e.status = : { $prefix } enabled " ;
$wheres [] = " { $prefix } ue.timestart < : { $prefix } now1 AND ( { $prefix } ue.timeend = 0 OR { $prefix } ue.timeend > : { $prefix } now2) " ;
$now = round ( time (), - 2 ); // rounding helps caching in DB
$params = array_merge ( $params , array ( $prefix . 'enabled' => ENROL_INSTANCE_ENABLED ,
$prefix . 'active' => ENROL_USER_ACTIVE ,
$prefix . 'now1' => $now , $prefix . 'now2' => $now ));
}
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
$joins = implode ( " \n " , $joins );
$wheres = " WHERE " . implode ( " AND " , $wheres );
$sql = " SELECT DISTINCT { $prefix } u.id
FROM { user } { $prefix } u
$joins
$wheres " ;
return array ( $sql , $params );
2010-03-31 07:41:31 +00:00
}
/**
2011-10-14 12:48:00 +02:00
* Returns list of users enrolled into course .
2010-03-31 07:41:31 +00:00
*
2012-01-14 19:12:43 +01:00
* @ package core_enrol
* @ category access
*
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ param string $withcapability
* @ param int $groupid 0 means ignore groups , any other value limits the result by group id
* @ param string $userfields requested user record fields
* @ param string $orderby
* @ param int $limitfrom return a subset of records , starting at this point ( optional , required if $limitnum is set ) .
* @ param int $limitnum return a subset comprising this many records ( optional , required if $limitfrom is set ) .
2013-03-27 10:14:10 +08:00
* @ param bool $onlyactive consider only active enrolments in enabled plugins and time restrictions
2011-10-14 12:48:00 +02:00
* @ return array of user records
2010-03-31 07:41:31 +00:00
*/
2013-03-27 10:14:10 +08:00
function get_enrolled_users ( context $context , $withcapability = '' , $groupid = 0 , $userfields = 'u.*' , $orderby = null ,
$limitfrom = 0 , $limitnum = 0 , $onlyactive = false ) {
2011-10-14 12:48:00 +02:00
global $DB ;
2010-09-17 07:39:48 +00:00
2013-03-27 10:14:10 +08:00
list ( $esql , $params ) = get_enrolled_sql ( $context , $withcapability , $groupid , $onlyactive );
2011-10-14 12:48:00 +02:00
$sql = " SELECT $userfields
FROM { user } u
JOIN ( $esql ) je ON je . id = u . id
WHERE u . deleted = 0 " ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
if ( $orderby ) {
$sql = " $sql ORDER BY $orderby " ;
2010-03-31 07:41:31 +00:00
} else {
2012-08-08 13:56:12 +01:00
list ( $sort , $sortparams ) = users_order_by_sql ( 'u' );
$sql = " $sql ORDER BY $sort " ;
$params = array_merge ( $params , $sortparams );
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
return $DB -> get_records_sql ( $sql , $params , $limitfrom , $limitnum );
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
/**
* Counts list of users enrolled into course ( as per above function )
*
2012-01-14 19:12:43 +01:00
* @ package core_enrol
* @ category access
*
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ param string $withcapability
* @ param int $groupid 0 means ignore groups , any other value limits the result by group id
2013-03-27 10:14:10 +08:00
* @ param bool $onlyactive consider only active enrolments in enabled plugins and time restrictions
2011-10-14 12:48:00 +02:00
* @ return array of user records
*/
2013-03-27 10:14:10 +08:00
function count_enrolled_users ( context $context , $withcapability = '' , $groupid = 0 , $onlyactive = false ) {
2011-10-14 12:48:00 +02:00
global $DB ;
2010-03-31 07:41:31 +00:00
2013-03-27 10:14:10 +08:00
list ( $esql , $params ) = get_enrolled_sql ( $context , $withcapability , $groupid , $onlyactive );
2011-10-14 12:48:00 +02:00
$sql = " SELECT count(u.id)
FROM { user } u
JOIN ( $esql ) je ON je . id = u . id
WHERE u . deleted = 0 " ;
return $DB -> count_records_sql ( $sql , $params );
}
/**
* Loads the capability definitions for the component ( from file ) .
*
* Loads the capability definitions for the component ( from file ) . If no
* capabilities are defined for the component , we simply return an empty array .
*
2012-01-14 19:12:43 +01:00
* @ access private
2011-10-14 12:48:00 +02:00
* @ param string $component full plugin name , examples : 'moodle' , 'mod_forum'
* @ return array array of capabilities
*/
function load_capability_def ( $component ) {
$defpath = get_component_directory ( $component ) . '/db/access.php' ;
$capabilities = array ();
if ( file_exists ( $defpath )) {
require ( $defpath );
if ( ! empty ( $ { $component . '_capabilities' })) {
// BC capability array name
// since 2.0 we prefer $capabilities instead - it is easier to use and matches db/* files
2011-10-25 14:20:22 +02:00
debugging ( 'componentname_capabilities array is deprecated, please use $capabilities array only in access.php files' );
2011-10-14 12:48:00 +02:00
$capabilities = $ { $component . '_capabilities' };
}
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
return $capabilities ;
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Gets the capabilities that have been cached in the database for this component .
*
2012-01-14 19:12:43 +01:00
* @ access private
2011-10-14 12:48:00 +02:00
* @ param string $component - examples : 'moodle' , 'mod_forum'
* @ return array array of capabilities
*/
function get_cached_capabilities ( $component = 'moodle' ) {
global $DB ;
return $DB -> get_records ( 'capabilities' , array ( 'component' => $component ));
}
2010-03-31 07:41:31 +00:00
/**
2011-10-14 12:48:00 +02:00
* Returns default capabilities for given role archetype .
2010-03-31 07:41:31 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param string $archetype role archetype
* @ return array
2010-03-31 07:41:31 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_default_capabilities ( $archetype ) {
global $DB ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
if ( ! $archetype ) {
return array ();
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
$alldefs = array ();
$defaults = array ();
$components = array ();
$allcaps = $DB -> get_records ( 'capabilities' );
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
foreach ( $allcaps as $cap ) {
if ( ! in_array ( $cap -> component , $components )) {
$components [] = $cap -> component ;
$alldefs = array_merge ( $alldefs , load_capability_def ( $cap -> component ));
}
}
foreach ( $alldefs as $name => $def ) {
// Use array 'archetypes if available. Only if not specified, use 'legacy'.
if ( isset ( $def [ 'archetypes' ])) {
if ( isset ( $def [ 'archetypes' ][ $archetype ])) {
$defaults [ $name ] = $def [ 'archetypes' ][ $archetype ];
}
// 'legacy' is for backward compatibility with 1.9 access.php
} else {
if ( isset ( $def [ 'legacy' ][ $archetype ])) {
$defaults [ $name ] = $def [ 'legacy' ][ $archetype ];
}
}
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
return $defaults ;
2010-03-31 07:41:31 +00:00
}
/**
2011-10-14 12:48:00 +02:00
* Reset role capabilities to default according to selected role archetype .
* If no archetype selected , removes all capabilities .
2010-03-31 07:41:31 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param int $roleid
* @ return void
2010-03-31 07:41:31 +00:00
*/
2011-10-14 12:48:00 +02:00
function reset_role_capabilities ( $roleid ) {
global $DB ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$role = $DB -> get_record ( 'role' , array ( 'id' => $roleid ), '*' , MUST_EXIST );
$defaultcaps = get_default_capabilities ( $role -> archetype );
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$systemcontext = context_system :: instance ();
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
2011-10-14 12:48:00 +02:00
$DB -> delete_records ( 'role_capabilities' , array ( 'roleid' => $roleid ));
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
foreach ( $defaultcaps as $cap => $permission ) {
assign_capability ( $cap , $permission , $roleid , $systemcontext -> id );
2010-03-31 07:41:31 +00:00
}
}
2010-11-11 03:37:56 +00:00
/**
2011-10-14 12:48:00 +02:00
* Updates the capabilities table with the component capability definitions .
* If no parameters are given , the function updates the core moodle
* capabilities .
2010-11-11 03:37:56 +00:00
*
2011-10-14 12:48:00 +02:00
* Note that the absence of the db / access . php capabilities definition file
* will cause any stored capabilities for the component to be removed from
* the database .
2010-11-11 03:37:56 +00:00
*
2012-01-14 19:12:43 +01:00
* @ access private
2011-10-14 12:48:00 +02:00
* @ param string $component examples : 'moodle' , 'mod/forum' , 'block/quiz_results'
* @ return boolean true if success , exception in case of any problems
2010-11-11 03:37:56 +00:00
*/
2011-10-14 12:48:00 +02:00
function update_capabilities ( $component = 'moodle' ) {
global $DB , $OUTPUT ;
2010-11-11 03:37:56 +00:00
2011-10-14 12:48:00 +02:00
$storedcaps = array ();
2010-11-11 03:37:56 +00:00
2011-10-14 12:48:00 +02:00
$filecaps = load_capability_def ( $component );
foreach ( $filecaps as $capname => $unused ) {
if ( ! preg_match ( '|^[a-z]+/[a-z_0-9]+:[a-z_0-9]+$|' , $capname )) {
debugging ( " Coding problem: Invalid capability name ' $capname ', use 'clonepermissionsfrom' field for migration. " );
}
2010-11-11 03:37:56 +00:00
}
2011-10-14 12:48:00 +02:00
$cachedcaps = get_cached_capabilities ( $component );
if ( $cachedcaps ) {
foreach ( $cachedcaps as $cachedcap ) {
array_push ( $storedcaps , $cachedcap -> name );
// update risk bitmasks and context levels in existing capabilities if needed
if ( array_key_exists ( $cachedcap -> name , $filecaps )) {
if ( ! array_key_exists ( 'riskbitmask' , $filecaps [ $cachedcap -> name ])) {
$filecaps [ $cachedcap -> name ][ 'riskbitmask' ] = 0 ; // no risk if not specified
}
if ( $cachedcap -> captype != $filecaps [ $cachedcap -> name ][ 'captype' ]) {
$updatecap = new stdClass ();
$updatecap -> id = $cachedcap -> id ;
$updatecap -> captype = $filecaps [ $cachedcap -> name ][ 'captype' ];
$DB -> update_record ( 'capabilities' , $updatecap );
}
if ( $cachedcap -> riskbitmask != $filecaps [ $cachedcap -> name ][ 'riskbitmask' ]) {
$updatecap = new stdClass ();
$updatecap -> id = $cachedcap -> id ;
$updatecap -> riskbitmask = $filecaps [ $cachedcap -> name ][ 'riskbitmask' ];
$DB -> update_record ( 'capabilities' , $updatecap );
}
2010-11-11 03:37:56 +00:00
2011-10-14 12:48:00 +02:00
if ( ! array_key_exists ( 'contextlevel' , $filecaps [ $cachedcap -> name ])) {
$filecaps [ $cachedcap -> name ][ 'contextlevel' ] = 0 ; // no context level defined
}
if ( $cachedcap -> contextlevel != $filecaps [ $cachedcap -> name ][ 'contextlevel' ]) {
$updatecap = new stdClass ();
$updatecap -> id = $cachedcap -> id ;
$updatecap -> contextlevel = $filecaps [ $cachedcap -> name ][ 'contextlevel' ];
$DB -> update_record ( 'capabilities' , $updatecap );
}
2010-11-11 03:37:56 +00:00
}
2011-10-14 12:48:00 +02:00
}
}
// Are there new capabilities in the file definition?
$newcaps = array ();
foreach ( $filecaps as $filecap => $def ) {
if ( ! $storedcaps ||
( $storedcaps && in_array ( $filecap , $storedcaps ) === false )) {
if ( ! array_key_exists ( 'riskbitmask' , $def )) {
$def [ 'riskbitmask' ] = 0 ; // no risk if not specified
2010-11-11 03:37:56 +00:00
}
2011-10-14 12:48:00 +02:00
$newcaps [ $filecap ] = $def ;
2010-11-11 03:37:56 +00:00
}
}
2011-10-14 12:48:00 +02:00
// Add new capabilities to the stored definition.
2012-04-19 08:47:06 +02:00
$existingcaps = $DB -> get_records_menu ( 'capabilities' , array (), 'id' , 'id, name' );
2011-10-14 12:48:00 +02:00
foreach ( $newcaps as $capname => $capdef ) {
$capability = new stdClass ();
$capability -> name = $capname ;
$capability -> captype = $capdef [ 'captype' ];
$capability -> contextlevel = $capdef [ 'contextlevel' ];
$capability -> component = $component ;
$capability -> riskbitmask = $capdef [ 'riskbitmask' ];
2010-11-11 03:37:56 +00:00
2011-10-14 12:48:00 +02:00
$DB -> insert_record ( 'capabilities' , $capability , false );
2012-04-19 08:47:06 +02:00
if ( isset ( $capdef [ 'clonepermissionsfrom' ]) && in_array ( $capdef [ 'clonepermissionsfrom' ], $existingcaps )){
2011-10-14 12:48:00 +02:00
if ( $rolecapabilities = $DB -> get_records ( 'role_capabilities' , array ( 'capability' => $capdef [ 'clonepermissionsfrom' ]))){
foreach ( $rolecapabilities as $rolecapability ){
//assign_capability will update rather than insert if capability exists
if ( ! assign_capability ( $capname , $rolecapability -> permission ,
$rolecapability -> roleid , $rolecapability -> contextid , true )){
echo $OUTPUT -> notification ( 'Could not clone capabilities for ' . $capname );
}
}
}
// we ignore archetype key if we have cloned permissions
} else if ( isset ( $capdef [ 'archetypes' ]) && is_array ( $capdef [ 'archetypes' ])) {
assign_legacy_capabilities ( $capname , $capdef [ 'archetypes' ]);
// 'legacy' is for backward compatibility with 1.9 access.php
} else if ( isset ( $capdef [ 'legacy' ]) && is_array ( $capdef [ 'legacy' ])) {
assign_legacy_capabilities ( $capname , $capdef [ 'legacy' ]);
2010-11-11 03:37:56 +00:00
}
}
2011-10-14 12:48:00 +02:00
// Are there any capabilities that have been removed from the file
// definition that we need to delete from the stored capabilities and
// role assignments?
capabilities_cleanup ( $component , $filecaps );
// reset static caches
accesslib_clear_all_caches ( false );
return true ;
2010-11-11 03:37:56 +00:00
}
2010-03-31 07:41:31 +00:00
/**
2011-10-14 12:48:00 +02:00
* Deletes cached capabilities that are no longer needed by the component .
* Also unassigns these capabilities from any roles that have them .
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
*
2012-01-14 19:12:43 +01:00
* @ access private
2011-10-14 12:48:00 +02:00
* @ param string $component examples : 'moodle' , 'mod_forum' , 'block_quiz_results'
* @ param array $newcapdef array of the new capability definitions that will be
* compared with the cached capabilities
* @ return int number of deprecated capabilities that have been removed
2010-03-31 07:41:31 +00:00
*/
2011-10-14 12:48:00 +02:00
function capabilities_cleanup ( $component , $newcapdef = null ) {
global $DB ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$removedcount = 0 ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
if ( $cachedcaps = get_cached_capabilities ( $component )) {
foreach ( $cachedcaps as $cachedcap ) {
if ( empty ( $newcapdef ) ||
array_key_exists ( $cachedcap -> name , $newcapdef ) === false ) {
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
// Remove from capabilities cache.
$DB -> delete_records ( 'capabilities' , array ( 'name' => $cachedcap -> name ));
$removedcount ++ ;
// Delete from roles.
if ( $roles = get_roles_with_capability ( $cachedcap -> name )) {
foreach ( $roles as $role ) {
if ( ! unassign_capability ( $cachedcap -> name , $role -> id )) {
print_error ( 'cannotunassigncap' , 'error' , '' , ( object ) array ( 'cap' => $cachedcap -> name , 'role' => $role -> name ));
}
}
}
} // End if.
}
}
return $removedcount ;
}
/**
* Returns an array of all the known types of risk
* The array keys can be used , for example as CSS class names , or in calls to
* print_risk_icon . The values are the corresponding RISK_ constants .
*
* @ return array all the known types of risk .
*/
function get_all_risks () {
return array (
'riskmanagetrust' => RISK_MANAGETRUST ,
'riskconfig' => RISK_CONFIG ,
'riskxss' => RISK_XSS ,
'riskpersonal' => RISK_PERSONAL ,
'riskspam' => RISK_SPAM ,
'riskdataloss' => RISK_DATALOSS ,
);
}
/**
* Return a link to moodle docs for a given capability name
*
2012-01-14 19:12:43 +01:00
* @ param stdClass $capability a capability - a row from the mdl_capabilities table .
2011-10-14 12:48:00 +02:00
* @ return string the human - readable capability name as a link to Moodle Docs .
*/
function get_capability_docs_link ( $capability ) {
$url = get_docs_url ( 'Capabilities/' . $capability -> name );
return '<a onclick="this.target=\'docspopup\'" href="' . $url . '">' . get_capability_string ( $capability -> name ) . '</a>' ;
}
/**
* This function pulls out all the resolved capabilities ( overrides and
* defaults ) of a role used in capability overrides in contexts at a given
* context .
*
* @ param int $roleid
2012-02-21 11:48:33 +13:00
* @ param context $context
2011-10-14 12:48:00 +02:00
* @ param string $cap capability , optional , defaults to ''
2012-02-21 11:48:33 +13:00
* @ return array Array of capabilities
2011-10-14 12:48:00 +02:00
*/
function role_context_capabilities ( $roleid , context $context , $cap = '' ) {
global $DB ;
$contexts = $context -> get_parent_context_ids ( true );
$contexts = '(' . implode ( ',' , $contexts ) . ')' ;
$params = array ( $roleid );
if ( $cap ) {
$search = " AND rc.capability = ? " ;
$params [] = $cap ;
} else {
$search = '' ;
}
$sql = " SELECT rc.*
FROM { role_capabilities } rc , { context } c
WHERE rc . contextid in $contexts
AND rc . roleid = ?
AND rc . contextid = c . id $search
ORDER BY c . contextlevel DESC , rc . capability DESC " ;
$capabilities = array ();
if ( $records = $DB -> get_records_sql ( $sql , $params )) {
// We are traversing via reverse order.
foreach ( $records as $record ) {
// If not set yet (i.e. inherit or not set at all), or currently we have a prohibit
if ( ! isset ( $capabilities [ $record -> capability ]) || $record -> permission <- 500 ) {
$capabilities [ $record -> capability ] = $record -> permission ;
}
}
}
return $capabilities ;
}
/**
* Constructs array with contextids as first parameter and context paths ,
* in both cases bottom top including self .
*
2012-01-14 19:12:43 +01:00
* @ access private
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ return array
*/
function get_context_info_list ( context $context ) {
$contextids = explode ( '/' , ltrim ( $context -> path , '/' ));
$contextpaths = array ();
$contextids2 = $contextids ;
while ( $contextids2 ) {
$contextpaths [] = '/' . implode ( '/' , $contextids2 );
array_pop ( $contextids2 );
}
return array ( $contextids , $contextpaths );
}
/**
* Check if context is the front page context or a context inside it
*
* Returns true if this context is the front page context , or a context inside it ,
* otherwise false .
*
* @ param context $context a context object .
* @ return bool
*/
function is_inside_frontpage ( context $context ) {
$frontpagecontext = context_course :: instance ( SITEID );
return strpos ( $context -> path . '/' , $frontpagecontext -> path . '/' ) === 0 ;
}
/**
* Returns capability information ( cached )
*
* @ param string $capabilityname
2012-01-14 19:12:43 +01:00
* @ return stdClass or null if capability not found
2011-10-14 12:48:00 +02:00
*/
function get_capability_info ( $capabilityname ) {
global $ACCESSLIB_PRIVATE , $DB ; // one request per page only
//TODO: MUC - this could be cached in shared memory, it would eliminate 1 query per page
if ( empty ( $ACCESSLIB_PRIVATE -> capabilities )) {
$ACCESSLIB_PRIVATE -> capabilities = array ();
$caps = $DB -> get_records ( 'capabilities' , array (), 'id, name, captype, riskbitmask' );
foreach ( $caps as $cap ) {
$capname = $cap -> name ;
unset ( $cap -> id );
unset ( $cap -> name );
$cap -> riskbitmask = ( int ) $cap -> riskbitmask ;
$ACCESSLIB_PRIVATE -> capabilities [ $capname ] = $cap ;
}
}
return isset ( $ACCESSLIB_PRIVATE -> capabilities [ $capabilityname ]) ? $ACCESSLIB_PRIVATE -> capabilities [ $capabilityname ] : null ;
}
/**
* Returns the human - readable , translated version of the capability .
* Basically a big switch statement .
*
* @ param string $capabilityname e . g . mod / choice : readresponses
* @ return string
*/
function get_capability_string ( $capabilityname ) {
// Typical capability name is 'plugintype/pluginname:capabilityname'
list ( $type , $name , $capname ) = preg_split ( '|[/:]|' , $capabilityname );
if ( $type === 'moodle' ) {
$component = 'core_role' ;
} else if ( $type === 'quizreport' ) {
//ugly hack!!
$component = 'quiz_' . $name ;
} else {
$component = $type . '_' . $name ;
}
$stringname = $name . ':' . $capname ;
if ( $component === 'core_role' or get_string_manager () -> string_exists ( $stringname , $component )) {
return get_string ( $stringname , $component );
}
$dir = get_component_directory ( $component );
if ( ! file_exists ( $dir )) {
// plugin broken or does not exist, do not bother with printing of debug message
return $capabilityname . ' ???' ;
}
// something is wrong in plugin, better print debug
return get_string ( $stringname , $component );
}
/**
* This gets the mod / block / course / core etc strings .
*
* @ param string $component
* @ param int $contextlevel
* @ return string | bool String is success , false if failed
*/
function get_component_string ( $component , $contextlevel ) {
if ( $component === 'moodle' or $component === 'core' ) {
switch ( $contextlevel ) {
// TODO: this should probably use context level names instead
case CONTEXT_SYSTEM : return get_string ( 'coresystem' );
case CONTEXT_USER : return get_string ( 'users' );
case CONTEXT_COURSECAT : return get_string ( 'categories' );
case CONTEXT_COURSE : return get_string ( 'course' );
case CONTEXT_MODULE : return get_string ( 'activities' );
case CONTEXT_BLOCK : return get_string ( 'block' );
default : print_error ( 'unknowncontext' );
}
}
list ( $type , $name ) = normalize_component ( $component );
$dir = get_plugin_directory ( $type , $name );
if ( ! file_exists ( $dir )) {
// plugin not installed, bad luck, there is no way to find the name
return $component . ' ???' ;
}
switch ( $type ) {
// TODO: this is really hacky, anyway it should be probably moved to lib/pluginlib.php
case 'quiz' : return get_string ( $name . ':componentname' , $component ); // insane hack!!!
case 'repository' : return get_string ( 'repository' , 'repository' ) . ': ' . get_string ( 'pluginname' , $component );
case 'gradeimport' : return get_string ( 'gradeimport' , 'grades' ) . ': ' . get_string ( 'pluginname' , $component );
case 'gradeexport' : return get_string ( 'gradeexport' , 'grades' ) . ': ' . get_string ( 'pluginname' , $component );
case 'gradereport' : return get_string ( 'gradereport' , 'grades' ) . ': ' . get_string ( 'pluginname' , $component );
case 'webservice' : return get_string ( 'webservice' , 'webservice' ) . ': ' . get_string ( 'pluginname' , $component );
case 'block' : return get_string ( 'block' ) . ': ' . get_string ( 'pluginname' , basename ( $component ));
case 'mod' :
if ( get_string_manager () -> string_exists ( 'pluginname' , $component )) {
return get_string ( 'activity' ) . ': ' . get_string ( 'pluginname' , $component );
} else {
return get_string ( 'activity' ) . ': ' . get_string ( 'modulename' , $component );
}
default : return get_string ( 'pluginname' , $component );
}
}
/**
* Gets the list of roles assigned to this context and up ( parents )
* from the list of roles that are visible on user profile page
* and participants page .
*
* @ param context $context
* @ return array
*/
function get_profile_roles ( context $context ) {
global $CFG , $DB ;
if ( empty ( $CFG -> profileroles )) {
return array ();
}
list ( $rallowed , $params ) = $DB -> get_in_or_equal ( explode ( ',' , $CFG -> profileroles ), SQL_PARAMS_NAMED , 'a' );
list ( $contextlist , $cparams ) = $DB -> get_in_or_equal ( $context -> get_parent_context_ids ( true ), SQL_PARAMS_NAMED , 'p' );
$params = array_merge ( $params , $cparams );
2012-05-18 11:40:35 +02:00
if ( $coursecontext = $context -> get_course_context ( false )) {
$params [ 'coursecontext' ] = $coursecontext -> id ;
} else {
$params [ 'coursecontext' ] = 0 ;
}
$sql = " SELECT DISTINCT r.id, r.name, r.shortname, r.sortorder, rn.name AS coursealias
2011-10-14 12:48:00 +02:00
FROM { role_assignments } ra , { role } r
2012-05-18 11:40:35 +02:00
LEFT JOIN { role_names } rn ON ( rn . contextid = : coursecontext AND rn . roleid = r . id )
2011-10-14 12:48:00 +02:00
WHERE r . id = ra . roleid
AND ra . contextid $contextlist
AND r . id $rallowed
ORDER BY r . sortorder ASC " ;
return $DB -> get_records_sql ( $sql , $params );
}
/**
* Gets the list of roles assigned to this context and up ( parents )
*
* @ param context $context
* @ return array
*/
function get_roles_used_in_context ( context $context ) {
global $DB ;
2012-05-18 11:40:35 +02:00
list ( $contextlist , $params ) = $DB -> get_in_or_equal ( $context -> get_parent_context_ids ( true ), SQL_PARAMS_NAMED , 'cl' );
if ( $coursecontext = $context -> get_course_context ( false )) {
$params [ 'coursecontext' ] = $coursecontext -> id ;
} else {
$params [ 'coursecontext' ] = 0 ;
}
2011-10-14 12:48:00 +02:00
2012-05-18 11:40:35 +02:00
$sql = " SELECT DISTINCT r.id, r.name, r.shortname, r.sortorder, rn.name AS coursealias
2011-10-14 12:48:00 +02:00
FROM { role_assignments } ra , { role } r
2012-05-18 11:40:35 +02:00
LEFT JOIN { role_names } rn ON ( rn . contextid = : coursecontext AND rn . roleid = r . id )
2011-10-14 12:48:00 +02:00
WHERE r . id = ra . roleid
AND ra . contextid $contextlist
ORDER BY r . sortorder ASC " ;
return $DB -> get_records_sql ( $sql , $params );
}
/**
* This function is used to print roles column in user profile page .
* It is using the CFG -> profileroles to limit the list to only interesting roles .
* ( The permission tab has full details of user role assignments . )
*
* @ param int $userid
* @ param int $courseid
* @ return string
*/
function get_user_roles_in_course ( $userid , $courseid ) {
global $CFG , $DB ;
if ( empty ( $CFG -> profileroles )) {
return '' ;
}
if ( $courseid == SITEID ) {
$context = context_system :: instance ();
} else {
$context = context_course :: instance ( $courseid );
}
if ( empty ( $CFG -> profileroles )) {
return array ();
}
list ( $rallowed , $params ) = $DB -> get_in_or_equal ( explode ( ',' , $CFG -> profileroles ), SQL_PARAMS_NAMED , 'a' );
list ( $contextlist , $cparams ) = $DB -> get_in_or_equal ( $context -> get_parent_context_ids ( true ), SQL_PARAMS_NAMED , 'p' );
$params = array_merge ( $params , $cparams );
2012-05-18 11:40:35 +02:00
if ( $coursecontext = $context -> get_course_context ( false )) {
$params [ 'coursecontext' ] = $coursecontext -> id ;
} else {
$params [ 'coursecontext' ] = 0 ;
}
$sql = " SELECT DISTINCT r.id, r.name, r.shortname, r.sortorder, rn.name AS coursealias
2011-10-14 12:48:00 +02:00
FROM { role_assignments } ra , { role } r
2012-05-18 11:40:35 +02:00
LEFT JOIN { role_names } rn ON ( rn . contextid = : coursecontext AND rn . roleid = r . id )
2011-10-14 12:48:00 +02:00
WHERE r . id = ra . roleid
AND ra . contextid $contextlist
AND r . id $rallowed
AND ra . userid = : userid
ORDER BY r . sortorder ASC " ;
$params [ 'userid' ] = $userid ;
$rolestring = '' ;
if ( $roles = $DB -> get_records_sql ( $sql , $params )) {
2012-05-18 11:40:35 +02:00
$rolenames = role_fix_names ( $roles , $context , ROLENAME_ALIAS , true ); // Substitute aliases
2011-10-14 12:48:00 +02:00
foreach ( $rolenames as $roleid => $rolename ) {
$rolenames [ $roleid ] = '<a href="' . $CFG -> wwwroot . '/user/index.php?contextid=' . $context -> id . '&roleid=' . $roleid . '">' . $rolename . '</a>' ;
}
$rolestring = implode ( ',' , $rolenames );
}
return $rolestring ;
}
/**
* Checks if a user can assign users to a particular role in this context
*
* @ param context $context
* @ param int $targetroleid - the id of the role you want to assign users to
* @ return boolean
*/
function user_can_assign ( context $context , $targetroleid ) {
global $DB ;
2012-12-06 10:52:09 +08:00
// First check to see if the user is a site administrator.
if ( is_siteadmin ()) {
return true ;
}
// Check if user has override capability.
// If not return false.
2011-10-14 12:48:00 +02:00
if ( ! has_capability ( 'moodle/role:assign' , $context )) {
return false ;
}
// pull out all active roles of this user from this context(or above)
if ( $userroles = get_user_roles ( $context )) {
foreach ( $userroles as $userrole ) {
// if any in the role_allow_override table, then it's ok
if ( $DB -> get_record ( 'role_allow_assign' , array ( 'roleid' => $userrole -> roleid , 'allowassign' => $targetroleid ))) {
return true ;
}
}
}
return false ;
}
/**
* Returns all site roles in correct sort order .
*
2012-05-18 11:40:35 +02:00
* @ param context $context optional context for course role name aliases
* @ return array of role records with optional coursealias property
2011-10-14 12:48:00 +02:00
*/
2012-05-18 11:40:35 +02:00
function get_all_roles ( context $context = null ) {
2011-10-14 12:48:00 +02:00
global $DB ;
2012-05-18 11:40:35 +02:00
if ( ! $context or ! $coursecontext = $context -> get_course_context ( false )) {
$coursecontext = null ;
}
if ( $coursecontext ) {
$sql = " SELECT r.*, rn.name AS coursealias
FROM { role } r
LEFT JOIN { role_names } rn ON ( rn . contextid = : coursecontext AND rn . roleid = r . id )
ORDER BY r . sortorder ASC " ;
return $DB -> get_records_sql ( $sql , array ( 'coursecontext' => $coursecontext -> id ));
} else {
return $DB -> get_records ( 'role' , array (), 'sortorder ASC' );
}
2011-10-14 12:48:00 +02:00
}
/**
* Returns roles of a specified archetype
*
* @ param string $archetype
* @ return array of full role records
*/
function get_archetype_roles ( $archetype ) {
global $DB ;
return $DB -> get_records ( 'role' , array ( 'archetype' => $archetype ), 'sortorder ASC' );
}
/**
* Gets all the user roles assigned in this context , or higher contexts
* this is mainly used when checking if a user can assign a role , or overriding a role
* i . e . we need to know what this user holds , in order to verify against allow_assign and
* allow_override tables
*
* @ param context $context
* @ param int $userid
* @ param bool $checkparentcontexts defaults to true
* @ param string $order defaults to 'c.contextlevel DESC, r.sortorder ASC'
* @ return array
*/
function get_user_roles ( context $context , $userid = 0 , $checkparentcontexts = true , $order = 'c.contextlevel DESC, r.sortorder ASC' ) {
global $USER , $DB ;
if ( empty ( $userid )) {
if ( empty ( $USER -> id )) {
return array ();
}
$userid = $USER -> id ;
}
if ( $checkparentcontexts ) {
$contextids = $context -> get_parent_context_ids ();
} else {
$contextids = array ();
}
$contextids [] = $context -> id ;
list ( $contextids , $params ) = $DB -> get_in_or_equal ( $contextids , SQL_PARAMS_QM );
array_unshift ( $params , $userid );
$sql = " SELECT ra.*, r.name, r.shortname
FROM { role_assignments } ra , { role } r , { context } c
WHERE ra . userid = ?
AND ra . roleid = r . id
AND ra . contextid = c . id
AND ra . contextid $contextids
ORDER BY $order " ;
return $DB -> get_records_sql ( $sql , $params );
}
2012-07-06 12:20:01 +01:00
/**
* Like get_user_roles , but adds in the authenticated user role , and the front
* page roles , if applicable .
*
* @ param context $context the context .
* @ param int $userid optional . Defaults to $USER -> id
* @ return array of objects with fields -> userid , -> contextid and -> roleid .
*/
function get_user_roles_with_special ( context $context , $userid = 0 ) {
global $CFG , $USER ;
if ( empty ( $userid )) {
if ( empty ( $USER -> id )) {
return array ();
}
$userid = $USER -> id ;
}
$ras = get_user_roles ( $context , $userid );
// Add front-page role if relevant.
$defaultfrontpageroleid = isset ( $CFG -> defaultfrontpageroleid ) ? $CFG -> defaultfrontpageroleid : 0 ;
$isfrontpage = ( $context -> contextlevel == CONTEXT_COURSE && $context -> instanceid == SITEID ) ||
is_inside_frontpage ( $context );
if ( $defaultfrontpageroleid && $isfrontpage ) {
$frontpagecontext = context_course :: instance ( SITEID );
$ra = new stdClass ();
$ra -> userid = $userid ;
$ra -> contextid = $frontpagecontext -> id ;
$ra -> roleid = $defaultfrontpageroleid ;
$ras [] = $ra ;
}
// Add authenticated user role if relevant.
$defaultuserroleid = isset ( $CFG -> defaultuserroleid ) ? $CFG -> defaultuserroleid : 0 ;
if ( $defaultuserroleid && ! isguestuser ( $userid )) {
$systemcontext = context_system :: instance ();
$ra = new stdClass ();
$ra -> userid = $userid ;
$ra -> contextid = $systemcontext -> id ;
$ra -> roleid = $defaultuserroleid ;
$ras [] = $ra ;
}
return $ras ;
}
2011-10-14 12:48:00 +02:00
/**
* Creates a record in the role_allow_override table
*
* @ param int $sroleid source roleid
* @ param int $troleid target roleid
* @ return void
*/
function allow_override ( $sroleid , $troleid ) {
global $DB ;
$record = new stdClass ();
$record -> roleid = $sroleid ;
$record -> allowoverride = $troleid ;
$DB -> insert_record ( 'role_allow_override' , $record );
}
/**
* Creates a record in the role_allow_assign table
*
* @ param int $fromroleid source roleid
* @ param int $targetroleid target roleid
* @ return void
*/
function allow_assign ( $fromroleid , $targetroleid ) {
global $DB ;
$record = new stdClass ();
$record -> roleid = $fromroleid ;
$record -> allowassign = $targetroleid ;
$DB -> insert_record ( 'role_allow_assign' , $record );
}
/**
* Creates a record in the role_allow_switch table
*
* @ param int $fromroleid source roleid
* @ param int $targetroleid target roleid
* @ return void
*/
function allow_switch ( $fromroleid , $targetroleid ) {
global $DB ;
$record = new stdClass ();
$record -> roleid = $fromroleid ;
$record -> allowswitch = $targetroleid ;
$DB -> insert_record ( 'role_allow_switch' , $record );
}
/**
* Gets a list of roles that this user can assign in this context
*
* @ param context $context the context .
* @ param int $rolenamedisplay the type of role name to display . One of the
* ROLENAME_X constants . Default ROLENAME_ALIAS .
* @ param bool $withusercounts if true , count the number of users with each role .
* @ param integer | object $user A user id or object . By default ( null ) checks the permissions of the current user .
* @ return array if $withusercounts is false , then an array $roleid => $rolename .
* if $withusercounts is true , returns a list of three arrays ,
* $rolenames , $rolecounts , and $nameswithcounts .
*/
function get_assignable_roles ( context $context , $rolenamedisplay = ROLENAME_ALIAS , $withusercounts = false , $user = null ) {
global $USER , $DB ;
// make sure there is a real user specified
if ( $user === null ) {
$userid = isset ( $USER -> id ) ? $USER -> id : 0 ;
} else {
$userid = is_object ( $user ) ? $user -> id : $user ;
}
if ( ! has_capability ( 'moodle/role:assign' , $context , $userid )) {
if ( $withusercounts ) {
return array ( array (), array (), array ());
} else {
return array ();
}
}
$params = array ();
$extrafields = '' ;
if ( $withusercounts ) {
$extrafields = ' , ( SELECT count ( u . id )
FROM { role_assignments } cra JOIN { user } u ON cra . userid = u . id
WHERE cra . roleid = r . id AND cra . contextid = : conid AND u . deleted = 0
) AS usercount ' ;
$params [ 'conid' ] = $context -> id ;
}
if ( is_siteadmin ( $userid )) {
// show all roles allowed in this context to admins
$assignrestriction = " " ;
} else {
2012-05-18 11:40:35 +02:00
$parents = $context -> get_parent_context_ids ( true );
$contexts = implode ( ',' , $parents );
2011-10-14 12:48:00 +02:00
$assignrestriction = " JOIN (SELECT DISTINCT raa.allowassign AS id
FROM { role_allow_assign } raa
JOIN { role_assignments } ra ON ra . roleid = raa . roleid
WHERE ra . userid = : userid AND ra . contextid IN ( $contexts )
) ar ON ar . id = r . id " ;
$params [ 'userid' ] = $userid ;
}
$params [ 'contextlevel' ] = $context -> contextlevel ;
2012-05-18 11:40:35 +02:00
if ( $coursecontext = $context -> get_course_context ( false )) {
$params [ 'coursecontext' ] = $coursecontext -> id ;
} else {
$params [ 'coursecontext' ] = 0 ; // no course aliases
$coursecontext = null ;
}
$sql = " SELECT r.id, r.name, r.shortname, rn.name AS coursealias $extrafields
2011-10-14 12:48:00 +02:00
FROM { role } r
$assignrestriction
2012-05-18 11:40:35 +02:00
JOIN { role_context_levels } rcl ON ( rcl . contextlevel = : contextlevel AND r . id = rcl . roleid )
LEFT JOIN { role_names } rn ON ( rn . contextid = : coursecontext AND rn . roleid = r . id )
2011-10-14 12:48:00 +02:00
ORDER BY r . sortorder ASC " ;
$roles = $DB -> get_records_sql ( $sql , $params );
2012-05-18 11:40:35 +02:00
$rolenames = role_fix_names ( $roles , $coursecontext , $rolenamedisplay , true );
2011-10-14 12:48:00 +02:00
if ( ! $withusercounts ) {
return $rolenames ;
}
$rolecounts = array ();
$nameswithcounts = array ();
foreach ( $roles as $role ) {
$nameswithcounts [ $role -> id ] = $rolenames [ $role -> id ] . ' (' . $roles [ $role -> id ] -> usercount . ')' ;
$rolecounts [ $role -> id ] = $roles [ $role -> id ] -> usercount ;
}
return array ( $rolenames , $rolecounts , $nameswithcounts );
}
/**
* Gets a list of roles that this user can switch to in a context
*
* Gets a list of roles that this user can switch to in a context , for the switchrole menu .
* This function just process the contents of the role_allow_switch table . You also need to
* test the moodle / role : switchroles to see if the user is allowed to switch in the first place .
*
* @ param context $context a context .
* @ return array an array $roleid => $rolename .
*/
function get_switchable_roles ( context $context ) {
global $USER , $DB ;
$params = array ();
$extrajoins = '' ;
$extrawhere = '' ;
if ( ! is_siteadmin ()) {
// Admins are allowed to switch to any role with.
// Others are subject to the additional constraint that the switch-to role must be allowed by
// 'role_allow_switch' for some role they have assigned in this context or any parent.
$parents = $context -> get_parent_context_ids ( true );
$contexts = implode ( ',' , $parents );
$extrajoins = " JOIN { role_allow_switch} ras ON ras.allowswitch = rc.roleid
JOIN { role_assignments } ra ON ra . roleid = ras . roleid " ;
$extrawhere = " WHERE ra.userid = :userid AND ra.contextid IN ( $contexts ) " ;
$params [ 'userid' ] = $USER -> id ;
}
2012-05-18 11:40:35 +02:00
if ( $coursecontext = $context -> get_course_context ( false )) {
$params [ 'coursecontext' ] = $coursecontext -> id ;
} else {
$params [ 'coursecontext' ] = 0 ; // no course aliases
$coursecontext = null ;
}
2011-10-14 12:48:00 +02:00
$query = "
2012-05-18 11:40:35 +02:00
SELECT r . id , r . name , r . shortname , rn . name AS coursealias
2011-10-14 12:48:00 +02:00
FROM ( SELECT DISTINCT rc . roleid
FROM { role_capabilities } rc
$extrajoins
$extrawhere ) idlist
JOIN { role } r ON r . id = idlist . roleid
2012-05-18 11:40:35 +02:00
LEFT JOIN { role_names } rn ON ( rn . contextid = : coursecontext AND rn . roleid = r . id )
2011-10-14 12:48:00 +02:00
ORDER BY r . sortorder " ;
2012-05-18 11:40:35 +02:00
$roles = $DB -> get_records_sql ( $query , $params );
2011-10-14 12:48:00 +02:00
2012-05-18 11:40:35 +02:00
return role_fix_names ( $roles , $context , ROLENAME_ALIAS , true );
2011-10-14 12:48:00 +02:00
}
/**
* Gets a list of roles that this user can override in this context .
*
* @ param context $context the context .
* @ param int $rolenamedisplay the type of role name to display . One of the
* ROLENAME_X constants . Default ROLENAME_ALIAS .
* @ param bool $withcounts if true , count the number of overrides that are set for each role .
* @ return array if $withcounts is false , then an array $roleid => $rolename .
* if $withusercounts is true , returns a list of three arrays ,
* $rolenames , $rolecounts , and $nameswithcounts .
*/
function get_overridable_roles ( context $context , $rolenamedisplay = ROLENAME_ALIAS , $withcounts = false ) {
global $USER , $DB ;
if ( ! has_any_capability ( array ( 'moodle/role:safeoverride' , 'moodle/role:override' ), $context )) {
if ( $withcounts ) {
return array ( array (), array (), array ());
} else {
return array ();
}
}
$parents = $context -> get_parent_context_ids ( true );
$contexts = implode ( ',' , $parents );
$params = array ();
$extrafields = '' ;
$params [ 'userid' ] = $USER -> id ;
if ( $withcounts ) {
$extrafields = ' , ( SELECT COUNT ( rc . id ) FROM { role_capabilities } rc
WHERE rc . roleid = ro . id AND rc . contextid = : conid ) AS overridecount ' ;
$params [ 'conid' ] = $context -> id ;
}
2012-05-18 11:40:35 +02:00
if ( $coursecontext = $context -> get_course_context ( false )) {
$params [ 'coursecontext' ] = $coursecontext -> id ;
} else {
$params [ 'coursecontext' ] = 0 ; // no course aliases
$coursecontext = null ;
}
2011-10-14 12:48:00 +02:00
if ( is_siteadmin ()) {
// show all roles to admins
$roles = $DB -> get_records_sql ( "
2012-05-18 11:40:35 +02:00
SELECT ro . id , ro . name , ro . shortname , rn . name AS coursealias $extrafields
2011-10-14 12:48:00 +02:00
FROM { role } ro
2012-05-18 11:40:35 +02:00
LEFT JOIN { role_names } rn ON ( rn . contextid = : coursecontext AND rn . roleid = ro . id )
2011-10-14 12:48:00 +02:00
ORDER BY ro . sortorder ASC " , $params );
} else {
$roles = $DB -> get_records_sql ( "
2012-05-18 11:40:35 +02:00
SELECT ro . id , ro . name , ro . shortname , rn . name AS coursealias $extrafields
2011-10-14 12:48:00 +02:00
FROM { role } ro
JOIN ( SELECT DISTINCT r . id
FROM { role } r
JOIN { role_allow_override } rao ON r . id = rao . allowoverride
JOIN { role_assignments } ra ON rao . roleid = ra . roleid
WHERE ra . userid = : userid AND ra . contextid IN ( $contexts )
) inline_view ON ro . id = inline_view . id
2012-05-18 11:40:35 +02:00
LEFT JOIN { role_names } rn ON ( rn . contextid = : coursecontext AND rn . roleid = ro . id )
2011-10-14 12:48:00 +02:00
ORDER BY ro . sortorder ASC " , $params );
}
2012-05-18 11:40:35 +02:00
$rolenames = role_fix_names ( $roles , $context , $rolenamedisplay , true );
2011-10-14 12:48:00 +02:00
if ( ! $withcounts ) {
return $rolenames ;
2012-05-18 11:40:35 +02:00
}
2011-10-14 12:48:00 +02:00
$rolecounts = array ();
$nameswithcounts = array ();
foreach ( $roles as $role ) {
$nameswithcounts [ $role -> id ] = $rolenames [ $role -> id ] . ' (' . $roles [ $role -> id ] -> overridecount . ')' ;
$rolecounts [ $role -> id ] = $roles [ $role -> id ] -> overridecount ;
}
return array ( $rolenames , $rolecounts , $nameswithcounts );
}
/**
* Create a role menu suitable for default role selection in enrol plugins .
2012-01-14 19:12:43 +01:00
*
* @ package core_enrol
*
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ param int $addroleid current or default role - always added to list
* @ return array roleid => localised role name
*/
function get_default_enrol_roles ( context $context , $addroleid = null ) {
global $DB ;
$params = array ( 'contextlevel' => CONTEXT_COURSE );
2012-05-18 11:40:35 +02:00
if ( $coursecontext = $context -> get_course_context ( false )) {
$params [ 'coursecontext' ] = $coursecontext -> id ;
} else {
$params [ 'coursecontext' ] = 0 ; // no course names
$coursecontext = null ;
}
2011-10-14 12:48:00 +02:00
if ( $addroleid ) {
$addrole = " OR r.id = :addroleid " ;
$params [ 'addroleid' ] = $addroleid ;
} else {
$addrole = " " ;
}
2012-05-18 11:40:35 +02:00
$sql = " SELECT r.id, r.name, r.shortname, rn.name AS coursealias
2011-10-14 12:48:00 +02:00
FROM { role } r
LEFT JOIN { role_context_levels } rcl ON ( rcl . roleid = r . id AND rcl . contextlevel = : contextlevel )
2012-05-18 11:40:35 +02:00
LEFT JOIN { role_names } rn ON ( rn . contextid = : coursecontext AND rn . roleid = r . id )
2011-10-14 12:48:00 +02:00
WHERE rcl . id IS NOT NULL $addrole
ORDER BY sortorder DESC " ;
2012-05-18 11:40:35 +02:00
$roles = $DB -> get_records_sql ( $sql , $params );
2011-10-14 12:48:00 +02:00
2012-05-18 11:40:35 +02:00
return role_fix_names ( $roles , $context , ROLENAME_BOTH , true );
2011-10-14 12:48:00 +02:00
}
/**
* Return context levels where this role is assignable .
2012-01-14 19:12:43 +01:00
*
2011-10-14 12:48:00 +02:00
* @ param integer $roleid the id of a role .
* @ return array list of the context levels at which this role may be assigned .
*/
function get_role_contextlevels ( $roleid ) {
global $DB ;
return $DB -> get_records_menu ( 'role_context_levels' , array ( 'roleid' => $roleid ),
'contextlevel' , 'id,contextlevel' );
}
/**
* Return roles suitable for assignment at the specified context level .
*
* NOTE : this function name looks like a typo , should be probably get_roles_for_contextlevel ()
*
* @ param integer $contextlevel a contextlevel .
* @ return array list of role ids that are assignable at this context level .
*/
function get_roles_for_contextlevels ( $contextlevel ) {
global $DB ;
return $DB -> get_records_menu ( 'role_context_levels' , array ( 'contextlevel' => $contextlevel ),
'' , 'id,roleid' );
}
/**
* Returns default context levels where roles can be assigned .
*
* @ param string $rolearchetype one of the role archetypes - that is , one of the keys
* from the array returned by get_role_archetypes ();
* @ return array list of the context levels at which this type of role may be assigned by default .
*/
function get_default_contextlevels ( $rolearchetype ) {
static $defaults = array (
'manager' => array ( CONTEXT_SYSTEM , CONTEXT_COURSECAT , CONTEXT_COURSE ),
'coursecreator' => array ( CONTEXT_SYSTEM , CONTEXT_COURSECAT ),
'editingteacher' => array ( CONTEXT_COURSE , CONTEXT_MODULE ),
'teacher' => array ( CONTEXT_COURSE , CONTEXT_MODULE ),
'student' => array ( CONTEXT_COURSE , CONTEXT_MODULE ),
'guest' => array (),
'user' => array (),
'frontpage' => array ());
if ( isset ( $defaults [ $rolearchetype ])) {
return $defaults [ $rolearchetype ];
} else {
return array ();
}
}
/**
* Set the context levels at which a particular role can be assigned .
* Throws exceptions in case of error .
*
* @ param integer $roleid the id of a role .
* @ param array $contextlevels the context levels at which this role should be assignable ,
* duplicate levels are removed .
* @ return void
*/
function set_role_contextlevels ( $roleid , array $contextlevels ) {
global $DB ;
$DB -> delete_records ( 'role_context_levels' , array ( 'roleid' => $roleid ));
$rcl = new stdClass ();
$rcl -> roleid = $roleid ;
$contextlevels = array_unique ( $contextlevels );
foreach ( $contextlevels as $level ) {
$rcl -> contextlevel = $level ;
$DB -> insert_record ( 'role_context_levels' , $rcl , false , true );
}
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
/**
* Who has this capability in this context ?
*
* This can be a very expensive call - use sparingly and keep
* the results if you are going to need them again soon .
*
* Note if $fields is empty this function attempts to get u .*
* which can get rather large - and has a serious perf impact
* on some DBs .
*
* @ param context $context
* @ param string | array $capability - capability name ( s )
* @ param string $fields - fields to be pulled . The user table is aliased to 'u' . u . id MUST be included .
* @ param string $sort - the sort order . Default is lastaccess time .
* @ param mixed $limitfrom - number of records to skip ( offset )
* @ param mixed $limitnum - number of records to fetch
* @ param string | array $groups - single group or array of groups - only return
* users who are in one of these group ( s ) .
* @ param string | array $exceptions - list of users to exclude , comma separated or array
* @ param bool $doanything_ignored not used any more , admin accounts are never returned
* @ param bool $view_ignored - use get_enrolled_sql () instead
* @ param bool $useviewallgroups if $groups is set the return users who
* have capability both $capability and moodle / site : accessallgroups
* in this context , as well as users who have $capability and who are
* in $groups .
2012-01-14 19:12:43 +01:00
* @ return array of user records
2011-10-14 12:48:00 +02:00
*/
function get_users_by_capability ( context $context , $capability , $fields = '' , $sort = '' , $limitfrom = '' , $limitnum = '' ,
$groups = '' , $exceptions = '' , $doanything_ignored = null , $view_ignored = null , $useviewallgroups = false ) {
global $CFG , $DB ;
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
2011-10-14 12:48:00 +02:00
$defaultuserroleid = isset ( $CFG -> defaultuserroleid ) ? $CFG -> defaultuserroleid : 0 ;
$defaultfrontpageroleid = isset ( $CFG -> defaultfrontpageroleid ) ? $CFG -> defaultfrontpageroleid : 0 ;
$ctxids = trim ( $context -> path , '/' );
$ctxids = str_replace ( '/' , ',' , $ctxids );
// Context is the frontpage
$iscoursepage = false ; // coursepage other than fp
$isfrontpage = false ;
if ( $context -> contextlevel == CONTEXT_COURSE ) {
if ( $context -> instanceid == SITEID ) {
$isfrontpage = true ;
} else {
$iscoursepage = true ;
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
}
$isfrontpage = ( $isfrontpage || is_inside_frontpage ( $context ));
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$caps = ( array ) $capability ;
// construct list of context paths bottom-->top
list ( $contextids , $paths ) = get_context_info_list ( $context );
// we need to find out all roles that have these capabilities either in definition or in overrides
$defs = array ();
list ( $incontexts , $params ) = $DB -> get_in_or_equal ( $contextids , SQL_PARAMS_NAMED , 'con' );
list ( $incaps , $params2 ) = $DB -> get_in_or_equal ( $caps , SQL_PARAMS_NAMED , 'cap' );
$params = array_merge ( $params , $params2 );
$sql = " SELECT rc.id, rc.roleid, rc.permission, rc.capability, ctx.path
FROM { role_capabilities } rc
JOIN { context } ctx on rc . contextid = ctx . id
WHERE rc . contextid $incontexts AND rc . capability $incaps " ;
$rcs = $DB -> get_records_sql ( $sql , $params );
foreach ( $rcs as $rc ) {
$defs [ $rc -> capability ][ $rc -> path ][ $rc -> roleid ] = $rc -> permission ;
}
// go through the permissions bottom-->top direction to evaluate the current permission,
// first one wins (prohibit is an exception that always wins)
$access = array ();
foreach ( $caps as $cap ) {
foreach ( $paths as $path ) {
if ( empty ( $defs [ $cap ][ $path ])) {
continue ;
}
foreach ( $defs [ $cap ][ $path ] as $roleid => $perm ) {
if ( $perm == CAP_PROHIBIT ) {
$access [ $cap ][ $roleid ] = CAP_PROHIBIT ;
2010-03-31 07:41:31 +00:00
continue ;
}
2011-10-14 12:48:00 +02:00
if ( ! isset ( $access [ $cap ][ $roleid ])) {
$access [ $cap ][ $roleid ] = ( int ) $perm ;
2010-03-31 07:41:31 +00:00
}
}
}
2011-10-14 12:48:00 +02:00
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
// make lists of roles that are needed and prohibited in this context
$needed = array (); // one of these is enough
$prohibited = array (); // must not have any of these
foreach ( $caps as $cap ) {
if ( empty ( $access [ $cap ])) {
continue ;
}
foreach ( $access [ $cap ] as $roleid => $perm ) {
2010-09-17 07:37:51 +00:00
if ( $perm == CAP_PROHIBIT ) {
2011-10-14 12:48:00 +02:00
unset ( $needed [ $cap ][ $roleid ]);
$prohibited [ $cap ][ $roleid ] = true ;
} else if ( $perm == CAP_ALLOW and empty ( $prohibited [ $cap ][ $roleid ])) {
$needed [ $cap ][ $roleid ] = true ;
2010-03-31 07:41:31 +00:00
}
}
2011-10-14 12:48:00 +02:00
if ( empty ( $needed [ $cap ]) or ! empty ( $prohibited [ $cap ][ $defaultuserroleid ])) {
// easy, nobody has the permission
unset ( $needed [ $cap ]);
unset ( $prohibited [ $cap ]);
} else if ( $isfrontpage and ! empty ( $prohibited [ $cap ][ $defaultfrontpageroleid ])) {
2012-01-14 19:12:43 +01:00
// everybody is disqualified on the frontpage
2011-10-14 12:48:00 +02:00
unset ( $needed [ $cap ]);
unset ( $prohibited [ $cap ]);
}
if ( empty ( $prohibited [ $cap ])) {
unset ( $prohibited [ $cap ]);
}
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
if ( empty ( $needed )) {
// there can not be anybody if no roles match this request
return array ();
}
if ( empty ( $prohibited )) {
// we can compact the needed roles
$n = array ();
foreach ( $needed as $cap ) {
foreach ( $cap as $roleid => $unused ) {
$n [ $roleid ] = true ;
}
}
$needed = array ( 'any' => $n );
unset ( $n );
}
2012-02-21 11:48:33 +13:00
// ***** Set up default fields ******
2011-10-14 12:48:00 +02:00
if ( empty ( $fields )) {
if ( $iscoursepage ) {
$fields = 'u.*, ul.timeaccess AS lastaccess' ;
} else {
$fields = 'u.*' ;
}
} else {
if ( debugging ( '' , DEBUG_DEVELOPER ) && strpos ( $fields , 'u.*' ) === false && strpos ( $fields , 'u.id' ) === false ) {
debugging ( 'u.id must be included in the list of fields passed to get_users_by_capability().' , DEBUG_DEVELOPER );
}
}
2012-02-21 11:48:33 +13:00
// Set up default sort
2011-10-14 12:48:00 +02:00
if ( empty ( $sort )) { // default to course lastaccess or just lastaccess
if ( $iscoursepage ) {
$sort = 'ul.timeaccess' ;
} else {
$sort = 'u.lastaccess' ;
}
}
// Prepare query clauses
$wherecond = array ();
$params = array ();
$joins = array ();
// User lastaccess JOIN
if (( strpos ( $sort , 'ul.timeaccess' ) === false ) and ( strpos ( $fields , 'ul.timeaccess' ) === false )) {
// user_lastaccess is not required MDL-13810
} else {
if ( $iscoursepage ) {
$joins [] = " LEFT OUTER JOIN { user_lastaccess} ul ON (ul.userid = u.id AND ul.courseid = { $context -> instanceid } ) " ;
} else {
throw new coding_exception ( 'Invalid sort in get_users_by_capability(), ul.timeaccess allowed only for course contexts.' );
}
}
2012-02-21 11:48:33 +13:00
// We never return deleted users or guest account.
2011-10-14 12:48:00 +02:00
$wherecond [] = " u.deleted = 0 AND u.id <> :guestid " ;
$params [ 'guestid' ] = $CFG -> siteguest ;
2012-02-21 11:48:33 +13:00
// Groups
2011-10-14 12:48:00 +02:00
if ( $groups ) {
$groups = ( array ) $groups ;
list ( $grouptest , $grpparams ) = $DB -> get_in_or_equal ( $groups , SQL_PARAMS_NAMED , 'grp' );
$grouptest = " u.id IN (SELECT userid FROM { groups_members} gm WHERE gm.groupid $grouptest ) " ;
$params = array_merge ( $params , $grpparams );
if ( $useviewallgroups ) {
$viewallgroupsusers = get_users_by_capability ( $context , 'moodle/site:accessallgroups' , 'u.id, u.id' , '' , '' , '' , '' , $exceptions );
if ( ! empty ( $viewallgroupsusers )) {
$wherecond [] = " ( $grouptest OR u.id IN ( " . implode ( ',' , array_keys ( $viewallgroupsusers )) . '))' ;
} else {
$wherecond [] = " ( $grouptest ) " ;
}
} else {
$wherecond [] = " ( $grouptest ) " ;
}
}
2012-02-21 11:48:33 +13:00
// User exceptions
2011-10-14 12:48:00 +02:00
if ( ! empty ( $exceptions )) {
$exceptions = ( array ) $exceptions ;
list ( $exsql , $exparams ) = $DB -> get_in_or_equal ( $exceptions , SQL_PARAMS_NAMED , 'exc' , false );
$params = array_merge ( $params , $exparams );
$wherecond [] = " u.id $exsql " ;
}
// now add the needed and prohibited roles conditions as joins
if ( ! empty ( $needed [ 'any' ])) {
// simple case - there are no prohibits involved
if ( ! empty ( $needed [ 'any' ][ $defaultuserroleid ]) or ( $isfrontpage and ! empty ( $needed [ 'any' ][ $defaultfrontpageroleid ]))) {
// everybody
} else {
$joins [] = " JOIN (SELECT DISTINCT userid
FROM { role_assignments }
WHERE contextid IN ( $ctxids )
AND roleid IN ( " .implode(',', array_keys( $needed['any'] )) . " )
) ra ON ra . userid = u . id " ;
}
} else {
$unions = array ();
$everybody = false ;
foreach ( $needed as $cap => $unused ) {
if ( empty ( $prohibited [ $cap ])) {
if ( ! empty ( $needed [ $cap ][ $defaultuserroleid ]) or ( $isfrontpage and ! empty ( $needed [ $cap ][ $defaultfrontpageroleid ]))) {
$everybody = true ;
break ;
} else {
$unions [] = " SELECT userid
FROM { role_assignments }
WHERE contextid IN ( $ctxids )
AND roleid IN ( " .implode(',', array_keys( $needed[$cap] )) . " ) " ;
}
} else {
if ( ! empty ( $prohibited [ $cap ][ $defaultuserroleid ]) or ( $isfrontpage and ! empty ( $prohibited [ $cap ][ $defaultfrontpageroleid ]))) {
// nobody can have this cap because it is prevented in default roles
continue ;
2007-08-06 18:45:35 +00:00
2011-10-14 12:48:00 +02:00
} else if ( ! empty ( $needed [ $cap ][ $defaultuserroleid ]) or ( $isfrontpage and ! empty ( $needed [ $cap ][ $defaultfrontpageroleid ]))) {
// everybody except the prohibitted - hiding does not matter
$unions [] = " SELECT id AS userid
FROM { user }
WHERE id NOT IN ( SELECT userid
FROM { role_assignments }
WHERE contextid IN ( $ctxids )
AND roleid IN ( " .implode(',', array_keys( $prohibited[$cap] )) . " )) " ;
2007-08-06 18:45:35 +00:00
2011-10-14 12:48:00 +02:00
} else {
$unions [] = " SELECT userid
FROM { role_assignments }
WHERE contextid IN ( $ctxids )
AND roleid IN ( " .implode(',', array_keys( $needed[$cap] )) . " )
AND roleid NOT IN ( " .implode(',', array_keys( $prohibited[$cap] )) . " ) " ;
}
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
}
if ( ! $everybody ) {
if ( $unions ) {
$joins [] = " JOIN (SELECT DISTINCT userid FROM ( " . implode ( ' UNION ' , $unions ) . " ) us) ra ON ra.userid = u.id " ;
} else {
// only prohibits found - nobody can be matched
$wherecond [] = " 1 = 2 " ;
2006-09-13 08:07:14 +00:00
}
2006-09-13 06:35:25 +00:00
}
2011-10-14 12:48:00 +02:00
}
2006-09-17 08:42:42 +00:00
2011-10-14 12:48:00 +02:00
// Collect WHERE conditions and needed joins
$where = implode ( ' AND ' , $wherecond );
if ( $where !== '' ) {
$where = 'WHERE ' . $where ;
}
$joins = implode ( " \n " , $joins );
2006-09-17 16:06:25 +00:00
2012-02-21 11:48:33 +13:00
// Ok, let's get the users!
2011-10-14 12:48:00 +02:00
$sql = " SELECT $fields
FROM { user } u
$joins
$where
ORDER BY $sort " ;
2006-09-17 16:06:25 +00:00
2011-10-14 12:48:00 +02:00
return $DB -> get_records_sql ( $sql , $params , $limitfrom , $limitnum );
}
2006-09-20 02:19:08 +00:00
2011-10-14 12:48:00 +02:00
/**
* Re - sort a users array based on a sorting policy
*
* Will re - sort a $users results array ( from get_users_by_capability (), usually )
* based on a sorting policy . This is to support the odd practice of
* sorting teachers by 'authority' , where authority was " lowest id of the role
* assignment " .
*
* Will execute 1 database query . Only suitable for small numbers of users , as it
* uses an u . id IN () clause .
*
* Notes about the sorting criteria .
*
* As a default , we cannot rely on role . sortorder because then
* admins / coursecreators will always win . That is why the sane
* rule " is locality matters most " , with sortorder as 2 nd
* consideration .
*
* If you want role . sortorder , use the 'sortorder' policy , and
* name explicitly what roles you want to cover . It ' s probably
* a good idea to see what roles have the capabilities you want
* ( array_diff () them against roiles that have 'can-do-anything'
* to weed out admin - ish roles . Or fetch a list of roles from
* variables like $CFG -> coursecontact .
*
* @ param array $users Users array , keyed on userid
* @ param context $context
* @ param array $roles ids of the roles to include , optional
* @ param string $sortpolicy defaults to locality , more about
* @ return array sorted copy of the array
*/
function sort_by_roleassignment_authority ( $users , context $context , $roles = array (), $sortpolicy = 'locality' ) {
global $DB ;
2006-09-20 02:19:08 +00:00
2011-10-14 12:48:00 +02:00
$userswhere = ' ra.userid IN (' . implode ( ',' , array_keys ( $users )) . ')' ;
$contextwhere = 'AND ra.contextid IN (' . str_replace ( '/' , ',' , substr ( $context -> path , 1 )) . ')' ;
if ( empty ( $roles )) {
$roleswhere = '' ;
2010-09-17 07:37:51 +00:00
} else {
2011-10-14 12:48:00 +02:00
$roleswhere = ' AND ra.roleid IN (' . implode ( ',' , $roles ) . ')' ;
2010-03-31 07:41:31 +00:00
}
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
$sql = " SELECT ra.userid
FROM { role_assignments } ra
JOIN { role } r
ON ra . roleid = r . id
JOIN { context } ctx
ON ra . contextid = ctx . id
WHERE $userswhere
$contextwhere
$roleswhere " ;
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
2011-10-14 12:48:00 +02:00
// Default 'locality' policy -- read PHPDoc notes
// about sort policies...
$orderby = 'ORDER BY '
. 'ctx.depth DESC, ' /* locality wins */
. 'r.sortorder ASC, ' /* rolesorting 2nd criteria */
. 'ra.id' ; /* role assignment order tie-breaker */
if ( $sortpolicy === 'sortorder' ) {
$orderby = 'ORDER BY '
. 'r.sortorder ASC, ' /* rolesorting 2nd criteria */
. 'ra.id' ; /* role assignment order tie-breaker */
2010-03-31 07:41:31 +00:00
}
2007-09-19 07:11:42 +00:00
2011-10-14 12:48:00 +02:00
$sortedids = $DB -> get_fieldset_sql ( $sql . $orderby );
$sortedusers = array ();
$seen = array ();
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
foreach ( $sortedids as $id ) {
// Avoid duplicates
if ( isset ( $seen [ $id ])) {
continue ;
}
$seen [ $id ] = true ;
2006-09-17 16:06:25 +00:00
2011-10-14 12:48:00 +02:00
// assign
$sortedusers [ $id ] = $users [ $id ];
}
return $sortedusers ;
2010-03-31 07:41:31 +00:00
}
/**
2011-10-14 12:48:00 +02:00
* Gets all the users assigned this role in this context or higher
*
* @ param int $roleid ( can also be an array of ints ! )
* @ param context $context
* @ param bool $parent if true , get list of users assigned in higher context too
* @ param string $fields fields from user ( u . ) , role assignment ( ra ) or role ( r . )
2012-08-08 13:56:12 +01:00
* @ param string $sort sort from user ( u . ) , role assignment ( ra . ) or role ( r . ) .
* null => use default sort from users_order_by_sql .
2012-12-06 15:26:43 +01:00
* @ param bool $all true means all , false means limit to enrolled users
2011-10-14 12:48:00 +02:00
* @ param string $group defaults to ''
* @ param mixed $limitfrom defaults to ''
* @ param mixed $limitnum defaults to ''
* @ param string $extrawheretest defaults to ''
2012-08-08 13:56:12 +01:00
* @ param array $whereorsortparams any paramter values used by $sort or $extrawheretest .
2011-10-14 12:48:00 +02:00
* @ return array
2010-03-31 07:41:31 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_role_users ( $roleid , context $context , $parent = false , $fields = '' ,
2012-12-06 15:26:43 +01:00
$sort = null , $all = true , $group = '' ,
2012-08-08 13:56:12 +01:00
$limitfrom = '' , $limitnum = '' , $extrawheretest = '' , $whereorsortparams = array ()) {
2010-03-31 07:41:31 +00:00
global $DB ;
2011-10-14 12:48:00 +02:00
if ( empty ( $fields )) {
$fields = 'u.id, u.confirmed, u.username, u.firstname, u.lastname, ' .
2011-11-16 13:01:55 +08:00
'u.maildisplay, u.mailformat, u.maildigest, u.email, u.emailstop, u.city, ' .
2011-10-14 12:48:00 +02:00
'u.country, u.picture, u.idnumber, u.department, u.institution, ' .
2012-05-18 11:40:35 +02:00
'u.lang, u.timezone, u.lastaccess, u.mnethostid, r.name AS rolename, r.sortorder, ' .
'r.shortname AS roleshortname, rn.name AS rolecoursealias' ;
2011-10-14 12:48:00 +02:00
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$parentcontexts = '' ;
if ( $parent ) {
$parentcontexts = substr ( $context -> path , 1 ); // kill leading slash
$parentcontexts = str_replace ( '/' , ',' , $parentcontexts );
if ( $parentcontexts !== '' ) {
$parentcontexts = ' OR ra.contextid IN (' . $parentcontexts . ' )' ;
}
}
if ( $roleid ) {
2012-05-18 11:40:35 +02:00
list ( $rids , $params ) = $DB -> get_in_or_equal ( $roleid , SQL_PARAMS_NAMED , 'r' );
2011-10-14 12:48:00 +02:00
$roleselect = " AND ra.roleid $rids " ;
2010-03-31 07:41:31 +00:00
} else {
2011-10-14 12:48:00 +02:00
$params = array ();
$roleselect = '' ;
2006-09-17 16:06:25 +00:00
}
2012-05-18 11:40:35 +02:00
if ( $coursecontext = $context -> get_course_context ( false )) {
$params [ 'coursecontext' ] = $coursecontext -> id ;
} else {
$params [ 'coursecontext' ] = 0 ;
}
2011-10-14 12:48:00 +02:00
if ( $group ) {
$groupjoin = " JOIN { groups_members} gm ON gm.userid = u.id " ;
2012-05-18 11:40:35 +02:00
$groupselect = " AND gm.groupid = :groupid " ;
$params [ 'groupid' ] = $group ;
2011-10-14 12:48:00 +02:00
} else {
$groupjoin = '' ;
$groupselect = '' ;
}
2006-09-17 16:06:25 +00:00
2012-05-18 11:40:35 +02:00
$params [ 'contextid' ] = $context -> id ;
2010-04-05 07:34:50 +00:00
2011-10-14 12:48:00 +02:00
if ( $extrawheretest ) {
$extrawheretest = ' AND ' . $extrawheretest ;
2012-08-08 13:56:12 +01:00
}
if ( $whereorsortparams ) {
2012-12-06 15:26:43 +01:00
$params = array_merge ( $params , $whereorsortparams );
2011-10-14 12:48:00 +02:00
}
2012-08-08 13:56:12 +01:00
if ( ! $sort ) {
list ( $sort , $sortparams ) = users_order_by_sql ( 'u' );
$params = array_merge ( $params , $sortparams );
}
2012-12-06 15:26:43 +01:00
if ( $all === null ) {
// Previously null was used to indicate that parameter was not used.
$all = true ;
}
if ( ! $all and $coursecontext ) {
// Do not use get_enrolled_sql() here for performance reasons.
$ejoin = " JOIN { user_enrolments} ue ON ue.userid = u.id
JOIN { enrol } e ON ( e . id = ue . enrolid AND e . courseid = : ecourseid ) " ;
$params [ 'ecourseid' ] = $coursecontext -> instanceid ;
} else {
$ejoin = " " ;
}
2011-10-14 12:48:00 +02:00
$sql = " SELECT DISTINCT $fields , ra.roleid
FROM { role_assignments } ra
JOIN { user } u ON u . id = ra . userid
JOIN { role } r ON ra . roleid = r . id
2012-12-06 15:26:43 +01:00
$ejoin
2012-05-18 11:40:35 +02:00
LEFT JOIN { role_names } rn ON ( rn . contextid = : coursecontext AND rn . roleid = r . id )
2011-10-14 12:48:00 +02:00
$groupjoin
2012-05-18 11:40:35 +02:00
WHERE ( ra . contextid = : contextid $parentcontexts )
2011-10-14 12:48:00 +02:00
$roleselect
$groupselect
$extrawheretest
ORDER BY $sort " ; // join now so that we can just use fullname() later
2010-04-05 07:34:50 +00:00
2011-10-14 12:48:00 +02:00
return $DB -> get_records_sql ( $sql , $params , $limitfrom , $limitnum );
2010-04-05 07:34:50 +00:00
}
2006-08-08 05:13:06 +00:00
/**
2011-10-14 12:48:00 +02:00
* Counts all the users assigned this role in this context or higher
2009-05-19 07:50:54 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param int | array $roleid either int or an array of ints
* @ param context $context
* @ param bool $parent if true , get list of users assigned in higher context too
* @ return int Returns the result count
2006-08-08 05:13:06 +00:00
*/
2011-10-14 12:48:00 +02:00
function count_role_users ( $roleid , context $context , $parent = false ) {
global $DB ;
2009-05-16 20:53:21 +00:00
2011-10-14 12:48:00 +02:00
if ( $parent ) {
if ( $contexts = $context -> get_parent_context_ids ()) {
$parentcontexts = ' OR r.contextid IN (' . implode ( ',' , $contexts ) . ')' ;
} else {
$parentcontexts = '' ;
2009-11-04 19:24:12 +00:00
}
2011-10-14 12:48:00 +02:00
} else {
$parentcontexts = '' ;
2006-08-08 05:13:06 +00:00
}
2009-06-19 14:25:56 +00:00
2011-10-14 12:48:00 +02:00
if ( $roleid ) {
list ( $rids , $params ) = $DB -> get_in_or_equal ( $roleid , SQL_PARAMS_QM );
$roleselect = " AND r.roleid $rids " ;
} else {
$params = array ();
$roleselect = '' ;
}
2006-08-08 05:13:06 +00:00
2011-10-14 12:48:00 +02:00
array_unshift ( $params , $context -> id );
2006-08-08 05:13:06 +00:00
2011-10-14 12:48:00 +02:00
$sql = " SELECT COUNT(u.id)
FROM { role_assignments } r
JOIN { user } u ON u . id = r . userid
WHERE ( r . contextid = ? $parentcontexts )
$roleselect
AND u . deleted = 0 " ;
return $DB -> count_records_sql ( $sql , $params );
2006-08-08 05:13:06 +00:00
}
2007-03-03 21:07:07 +00:00
/**
2011-10-14 12:48:00 +02:00
* This function gets the list of courses that this user has a particular capability in .
* It is still not very efficient .
*
* @ param string $capability Capability in question
* @ param int $userid User ID or null for current user
* @ param bool $doanything True if 'doanything' is permitted ( default )
* @ param string $fieldsexceptid Leave blank if you only need 'id' in the course records ;
* otherwise use a comma - separated list of the fields you require , not including id
* @ param string $orderby If set , use a comma - separated list of fields from course
* table with sql modifiers ( DESC ) if needed
* @ return array Array of courses , may have zero entries . Or false if query failed .
2007-03-03 21:07:07 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_user_capability_course ( $capability , $userid = null , $doanything = true , $fieldsexceptid = '' , $orderby = '' ) {
2008-05-24 18:35:48 +00:00
global $DB ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
// Convert fields list and ordering
$fieldlist = '' ;
if ( $fieldsexceptid ) {
$fields = explode ( ',' , $fieldsexceptid );
foreach ( $fields as $field ) {
$fieldlist .= ',c.' . $field ;
2007-03-03 21:07:07 +00:00
}
}
2011-10-14 12:48:00 +02:00
if ( $orderby ) {
$fields = explode ( ',' , $orderby );
$orderby = '' ;
foreach ( $fields as $field ) {
if ( $orderby ) {
$orderby .= ',' ;
2010-05-21 11:43:37 +00:00
}
2011-10-14 12:48:00 +02:00
$orderby .= 'c.' . $field ;
2007-03-03 21:07:07 +00:00
}
2011-10-14 12:48:00 +02:00
$orderby = 'ORDER BY ' . $orderby ;
2007-03-03 21:07:07 +00:00
}
2011-10-14 12:48:00 +02:00
// Obtain a list of everything relevant about all courses including context.
// Note the result can be used directly as a context (we are going to), the course
// fields are just appended.
2012-01-14 21:36:02 +01:00
$contextpreload = context_helper :: get_preload_record_columns_sql ( 'x' );
2011-10-14 12:48:00 +02:00
$courses = array ();
2012-01-14 21:36:02 +01:00
$rs = $DB -> get_recordset_sql ( " SELECT c.id $fieldlist , $contextpreload
2011-10-14 12:48:00 +02:00
FROM { course } c
2012-01-14 21:36:02 +01:00
JOIN { context } x ON ( c . id = x . instanceid AND x . contextlevel = " .CONTEXT_COURSE. " )
2011-10-14 12:48:00 +02:00
$orderby " );
// Check capability for each course in turn
2012-01-14 21:36:02 +01:00
foreach ( $rs as $course ) {
context_helper :: preload_from_record ( $course );
$context = context_course :: instance ( $course -> id );
if ( has_capability ( $capability , $context , $userid , $doanything )) {
2011-10-14 12:48:00 +02:00
// We've got the capability. Make the record look like a course record
// and store it
2012-01-14 21:36:02 +01:00
$courses [] = $course ;
2011-10-14 12:48:00 +02:00
}
}
$rs -> close ();
return empty ( $courses ) ? false : $courses ;
2007-03-03 21:07:07 +00:00
}
2006-08-08 05:13:06 +00:00
2007-03-05 11:27:01 +00:00
/**
2011-10-14 12:48:00 +02:00
* This function finds the roles assigned directly to this context only
* i . e . no roles in parent contexts
*
* @ param context $context
* @ return array
2007-03-05 11:27:01 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_roles_on_exact_context ( context $context ) {
2008-05-24 18:35:48 +00:00
global $DB ;
2011-10-14 12:48:00 +02:00
return $DB -> get_records_sql ( " SELECT r.*
FROM { role_assignments } ra , { role } r
WHERE ra . roleid = r . id AND ra . contextid = ? " ,
array ( $context -> id ));
2007-03-05 11:27:01 +00:00
}
2006-08-08 05:13:06 +00:00
/**
2011-10-14 12:48:00 +02:00
* Switches the current user to another role for the current session and only
* in the given context .
2006-08-08 05:13:06 +00:00
*
2011-10-14 12:48:00 +02:00
* The caller * must * check
* - that this op is allowed
* - that the requested role can be switched to in this context ( use get_switchable_roles )
* - that the requested role is NOT $CFG -> defaultuserroleid
2006-08-08 05:13:06 +00:00
*
2011-10-14 12:48:00 +02:00
* To " unswitch " pass 0 as the roleid .
*
* This function * will * modify $USER -> access - beware
*
* @ param integer $roleid the role to switch to .
* @ param context $context the context in which to perform the switch .
* @ return bool success or failure .
2006-08-08 05:13:06 +00:00
*/
2011-10-14 12:48:00 +02:00
function role_switch ( $roleid , context $context ) {
global $USER ;
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
//
// Plan of action
//
// - Add the ghost RA to $USER->access
// as $USER->access['rsw'][$path] = $roleid
//
// - Make sure $USER->access['rdef'] has the roledefs
// it needs to honour the switcherole
//
// Roledefs will get loaded "deep" here - down to the last child
// context. Note that
//
// - When visiting subcontexts, our selective accessdata loading
// will still work fine - though those ra/rdefs will be ignored
// appropriately while the switch is in place
//
// - If a switcherole happens at a category with tons of courses
// (that have many overrides for switched-to role), the session
// will get... quite large. Sometimes you just can't win.
//
// To un-switch just unset($USER->access['rsw'][$path])
//
// Note: it is not possible to switch to roles that do not have course:view
2006-09-20 21:00:45 +00:00
2012-01-15 00:14:44 +01:00
if ( ! isset ( $USER -> access )) {
load_all_capabilities ();
2011-10-14 12:48:00 +02:00
}
2007-09-13 13:44:35 +00:00
2012-01-15 00:14:44 +01:00
// Add the switch RA
2011-10-14 12:48:00 +02:00
if ( $roleid == 0 ) {
unset ( $USER -> access [ 'rsw' ][ $context -> path ]);
return true ;
2006-08-08 05:13:06 +00:00
}
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
$USER -> access [ 'rsw' ][ $context -> path ] = $roleid ;
// Load roledefs
load_role_access_by_context ( $roleid , $context , $USER -> access );
2008-12-12 10:50:25 +00:00
2006-08-08 05:13:06 +00:00
return true ;
}
2011-10-14 12:48:00 +02:00
/**
* Checks if the user has switched roles within the given course .
*
2012-01-14 19:12:43 +01:00
* Note : You can only switch roles within the course , hence it takes a course id
2011-10-14 12:48:00 +02:00
* rather than a context . On that note Petr volunteered to implement this across
* all other contexts , all requests for this should be forwarded to him ;)
*
* @ param int $courseid The id of the course to check
* @ return bool True if the user has switched roles within the course .
*/
function is_role_switched ( $courseid ) {
global $USER ;
$context = context_course :: instance ( $courseid , MUST_EXIST );
return ( ! empty ( $USER -> access [ 'rsw' ][ $context -> path ]));
}
2006-08-08 05:13:06 +00:00
/**
2011-10-14 12:48:00 +02:00
* Get any role that has an override on exact context
2009-05-22 02:03:55 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param context $context The context to check
* @ return array An array of roles
2006-08-08 05:13:06 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_roles_with_override_on_context ( context $context ) {
2008-05-24 18:35:48 +00:00
global $DB ;
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
return $DB -> get_records_sql ( " SELECT r.*
FROM { role_capabilities } rc , { role } r
WHERE rc . roleid = r . id AND rc . contextid = ? " ,
array ( $context -> id ));
}
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
/**
* Get all capabilities for this role on this context ( overrides )
*
* @ param stdClass $role
* @ param context $context
* @ return array
*/
function get_capabilities_from_role_on_context ( $role , context $context ) {
global $DB ;
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
return $DB -> get_records_sql ( " SELECT *
FROM { role_capabilities }
WHERE contextid = ? AND roleid = ? " ,
array ( $context -> id , $role -> id ));
2006-08-08 05:13:06 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Find out which roles has assignment on this context
*
* @ param context $context
* @ return array
*
*/
function get_roles_with_assignment_on_context ( context $context ) {
global $DB ;
return $DB -> get_records_sql ( " SELECT r.*
FROM { role_assignments } ra , { role } r
WHERE ra . roleid = r . id AND ra . contextid = ? " ,
array ( $context -> id ));
}
2006-08-08 05:13:06 +00:00
2011-10-14 12:48:00 +02:00
/**
* Find all user assignment of users for this role , on this context
*
* @ param stdClass $role
* @ param context $context
* @ return array
*/
function get_users_from_role_on_context ( $role , context $context ) {
global $DB ;
2006-08-08 05:13:06 +00:00
2011-10-14 12:48:00 +02:00
return $DB -> get_records_sql ( " SELECT *
FROM { role_assignments }
WHERE contextid = ? AND roleid = ? " ,
array ( $context -> id , $role -> id ));
}
2006-08-08 05:13:06 +00:00
2008-11-21 06:19:52 +00:00
/**
2011-10-14 12:48:00 +02:00
* Simple function returning a boolean true if user has roles
* in context or parent contexts , otherwise false .
*
* @ param int $userid
* @ param int $roleid
* @ param int $contextid empty means any context
* @ return bool
2008-11-21 06:19:52 +00:00
*/
2011-10-14 12:48:00 +02:00
function user_has_role_assignment ( $userid , $roleid , $contextid = 0 ) {
global $DB ;
if ( $contextid ) {
if ( ! $context = context :: instance_by_id ( $contextid , IGNORE_MISSING )) {
return false ;
}
$parents = $context -> get_parent_context_ids ( true );
list ( $contexts , $params ) = $DB -> get_in_or_equal ( $parents , SQL_PARAMS_NAMED , 'r' );
$params [ 'userid' ] = $userid ;
$params [ 'roleid' ] = $roleid ;
$sql = " SELECT COUNT(ra.id)
FROM { role_assignments } ra
WHERE ra . userid = : userid AND ra . roleid = : roleid AND ra . contextid $contexts " ;
$count = $DB -> get_field_sql ( $sql , $params );
return ( $count > 0 );
} else {
return $DB -> record_exists ( 'role_assignments' , array ( 'userid' => $userid , 'roleid' => $roleid ));
2008-11-21 06:19:52 +00:00
}
}
2006-08-08 05:13:06 +00:00
/**
2011-10-14 12:48:00 +02:00
* Get role name or alias if exists and format the text .
2008-10-30 06:26:18 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param stdClass $role role object
2012-05-18 11:40:35 +02:00
* - optional 'coursealias' property should be included for performance reasons if course context used
* - description property is not required here
* @ param context | bool $context empty means system context
* @ param int $rolenamedisplay type of role name
* @ return string localised role name or course role name alias
2006-08-08 05:13:06 +00:00
*/
2012-05-18 11:40:35 +02:00
function role_get_name ( stdClass $role , $context = null , $rolenamedisplay = ROLENAME_ALIAS ) {
2008-05-24 18:35:48 +00:00
global $DB ;
2006-08-13 15:48:57 +00:00
2012-05-18 11:40:35 +02:00
if ( $rolenamedisplay == ROLENAME_SHORT ) {
return $role -> shortname ;
}
if ( ! $context or ! $coursecontext = $context -> get_course_context ( false )) {
$coursecontext = null ;
}
if ( $coursecontext and ! property_exists ( $role , 'coursealias' ) and ( $rolenamedisplay == ROLENAME_ALIAS or $rolenamedisplay == ROLENAME_BOTH or $rolenamedisplay == ROLENAME_ALIAS_RAW )) {
$role = clone ( $role ); // Do not modify parameters.
if ( $r = $DB -> get_record ( 'role_names' , array ( 'roleid' => $role -> id , 'contextid' => $coursecontext -> id ))) {
$role -> coursealias = $r -> name ;
} else {
$role -> coursealias = null ;
}
}
if ( $rolenamedisplay == ROLENAME_ALIAS_RAW ) {
if ( $coursecontext ) {
return $role -> coursealias ;
} else {
return null ;
}
}
if ( trim ( $role -> name ) !== '' ) {
// For filtering always use context where was the thing defined - system for roles here.
$original = format_string ( $role -> name , true , array ( 'context' => context_system :: instance ()));
2011-10-14 12:48:00 +02:00
} else {
2012-05-18 11:40:35 +02:00
// Empty role->name means we want to see localised role name based on shortname,
// only default roles are supposed to be localised.
switch ( $role -> shortname ) {
case 'manager' : $original = get_string ( 'manager' , 'role' ); break ;
case 'coursecreator' : $original = get_string ( 'coursecreators' ); break ;
case 'editingteacher' : $original = get_string ( 'defaultcourseteacher' ); break ;
case 'teacher' : $original = get_string ( 'noneditingteacher' ); break ;
case 'student' : $original = get_string ( 'defaultcoursestudent' ); break ;
case 'guest' : $original = get_string ( 'guest' ); break ;
case 'user' : $original = get_string ( 'authenticateduser' ); break ;
case 'frontpage' : $original = get_string ( 'frontpageuser' , 'role' ); break ;
// We should not get here, the role UI should require the name for custom roles!
default : $original = $role -> shortname ; break ;
}
}
if ( $rolenamedisplay == ROLENAME_ORIGINAL ) {
return $original ;
}
if ( $rolenamedisplay == ROLENAME_ORIGINALANDSHORT ) {
return " $original ( $role->shortname ) " ;
}
if ( $rolenamedisplay == ROLENAME_ALIAS ) {
if ( $coursecontext and trim ( $role -> coursealias ) !== '' ) {
return format_string ( $role -> coursealias , true , array ( 'context' => $coursecontext ));
} else {
return $original ;
}
}
if ( $rolenamedisplay == ROLENAME_BOTH ) {
if ( $coursecontext and trim ( $role -> coursealias ) !== '' ) {
return format_string ( $role -> coursealias , true , array ( 'context' => $coursecontext )) . " ( $original ) " ;
} else {
return $original ;
}
}
throw new coding_exception ( 'Invalid $rolenamedisplay parameter specified in role_get_name()' );
}
/**
* Returns localised role description if available .
* If the name is empty it tries to find the default role name using
* hardcoded list of default role names or other methods in the future .
*
* @ param stdClass $role
* @ return string localised role name
*/
function role_get_description ( stdClass $role ) {
if ( ! html_is_blank ( $role -> description )) {
return format_text ( $role -> description , FORMAT_HTML , array ( 'context' => context_system :: instance ()));
}
switch ( $role -> shortname ) {
case 'manager' : return get_string ( 'managerdescription' , 'role' );
case 'coursecreator' : return get_string ( 'coursecreatorsdescription' );
case 'editingteacher' : return get_string ( 'defaultcourseteacherdescription' );
case 'teacher' : return get_string ( 'noneditingteacherdescription' );
case 'student' : return get_string ( 'defaultcoursestudentdescription' );
case 'guest' : return get_string ( 'guestdescription' );
case 'user' : return get_string ( 'authenticateduserdescription' );
case 'frontpage' : return get_string ( 'frontpageuserdescription' , 'role' );
default : return '' ;
2011-10-14 12:48:00 +02:00
}
}
2006-08-08 05:13:06 +00:00
2012-07-06 12:20:01 +01:00
/**
* Get all the localised role names for a context .
2013-03-17 20:15:59 +01:00
*
* @ param context $context the context , null means system context
2012-07-06 12:20:01 +01:00
* @ param array of role objects with a -> localname field containing the context - specific role name .
2013-03-17 20:15:59 +01:00
* @ param int $rolenamedisplay
* @ param bool $returnmenu true means id => localname , false means id => rolerecord
* @ return array Array of context - specific role names , or role objects with a -> localname field added .
2012-07-06 12:20:01 +01:00
*/
2013-03-17 20:15:59 +01:00
function role_get_names ( context $context = null , $rolenamedisplay = ROLENAME_ALIAS , $returnmenu = null ) {
return role_fix_names ( get_all_roles ( $context ), $context , $rolenamedisplay , $returnmenu );
2012-07-06 12:20:01 +01:00
}
2011-10-14 12:48:00 +02:00
/**
* Prepare list of roles for display , apply aliases and format text
*
2012-05-18 11:40:35 +02:00
* @ param array $roleoptions array roleid => roleobject ( with optional coursealias ), strings are accepted for backwards compatibility only
2013-03-17 20:15:59 +01:00
* @ param context $context the context , null means system context
2011-10-14 12:48:00 +02:00
* @ param int $rolenamedisplay
2012-05-18 11:40:35 +02:00
* @ param bool $returnmenu null means keep the same format as $roleoptions , true means id => localname , false means id => rolerecord
2012-01-14 19:12:43 +01:00
* @ return array Array of context - specific role names , or role objects with a -> localname field added .
2011-10-14 12:48:00 +02:00
*/
2013-03-17 20:15:59 +01:00
function role_fix_names ( $roleoptions , context $context = null , $rolenamedisplay = ROLENAME_ALIAS , $returnmenu = null ) {
2011-10-14 12:48:00 +02:00
global $DB ;
2006-08-13 15:48:57 +00:00
2012-05-18 11:40:35 +02:00
if ( empty ( $roleoptions )) {
return array ();
}
2006-08-08 05:13:06 +00:00
2012-05-18 11:40:35 +02:00
if ( ! $context or ! $coursecontext = $context -> get_course_context ( false )) {
$coursecontext = null ;
}
// We usually need all role columns...
$first = reset ( $roleoptions );
if ( $returnmenu === null ) {
$returnmenu = ! is_object ( $first );
}
if ( ! is_object ( $first ) or ! property_exists ( $first , 'shortname' )) {
$allroles = get_all_roles ( $context );
foreach ( $roleoptions as $rid => $unused ) {
$roleoptions [ $rid ] = $allroles [ $rid ];
2011-10-14 12:48:00 +02:00
}
}
2006-08-08 05:13:06 +00:00
2012-05-18 11:40:35 +02:00
// Inject coursealias if necessary.
if ( $coursecontext and ( $rolenamedisplay == ROLENAME_ALIAS_RAW or $rolenamedisplay == ROLENAME_ALIAS or $rolenamedisplay == ROLENAME_BOTH )) {
$first = reset ( $roleoptions );
if ( ! property_exists ( $first , 'coursealias' )) {
$aliasnames = $DB -> get_records ( 'role_names' , array ( 'contextid' => $coursecontext -> id ));
foreach ( $aliasnames as $alias ) {
if ( isset ( $roleoptions [ $alias -> roleid ])) {
$roleoptions [ $alias -> roleid ] -> coursealias = $alias -> name ;
2006-08-09 04:51:05 +00:00
}
}
2011-10-14 12:48:00 +02:00
}
2006-08-13 15:48:57 +00:00
}
2008-10-30 06:26:18 +00:00
2012-05-18 11:40:35 +02:00
// Add localname property.
foreach ( $roleoptions as $rid => $role ) {
$roleoptions [ $rid ] -> localname = role_get_name ( $role , $coursecontext , $rolenamedisplay );
}
if ( ! $returnmenu ) {
return $roleoptions ;
2011-10-14 12:48:00 +02:00
}
2012-05-18 11:40:35 +02:00
$menu = array ();
foreach ( $roleoptions as $rid => $role ) {
$menu [ $rid ] = $role -> localname ;
}
return $menu ;
2006-08-08 05:13:06 +00:00
}
2008-10-30 06:26:18 +00:00
/**
2011-10-14 12:48:00 +02:00
* Aids in detecting if a new line is required when reading a new capability
2008-10-30 06:26:18 +00:00
*
2011-10-14 12:48:00 +02:00
* This function helps admin / roles / manage . php etc to detect if a new line should be printed
* when we read in a new capability .
* Most of the time , if the 2 components are different we should print a new line , ( e . g . course system -> rss client )
* but when we are in grade , all reports / import / export capabilities should be together
*
* @ param string $cap component string a
* @ param string $comp component string b
* @ param int $contextlevel
* @ return bool whether 2 component are in different " sections "
*/
function component_level_changed ( $cap , $comp , $contextlevel ) {
2008-10-30 06:26:18 +00:00
2011-10-14 12:48:00 +02:00
if ( strstr ( $cap -> component , '/' ) && strstr ( $comp , '/' )) {
$compsa = explode ( '/' , $cap -> component );
$compsb = explode ( '/' , $comp );
2008-10-30 06:26:18 +00:00
2011-10-14 12:48:00 +02:00
// list of system reports
if (( $compsa [ 0 ] == 'report' ) && ( $compsb [ 0 ] == 'report' )) {
return false ;
}
2008-10-30 06:26:18 +00:00
2011-10-14 12:48:00 +02:00
// we are in gradebook, still
if (( $compsa [ 0 ] == 'gradeexport' || $compsa [ 0 ] == 'gradeimport' || $compsa [ 0 ] == 'gradereport' ) &&
( $compsb [ 0 ] == 'gradeexport' || $compsb [ 0 ] == 'gradeimport' || $compsb [ 0 ] == 'gradereport' )) {
return false ;
}
2008-10-30 06:26:18 +00:00
2011-10-14 12:48:00 +02:00
if (( $compsa [ 0 ] == 'coursereport' ) && ( $compsb [ 0 ] == 'coursereport' )) {
return false ;
}
2008-10-30 06:26:18 +00:00
}
2011-10-14 12:48:00 +02:00
return ( $cap -> component != $comp || $cap -> contextlevel != $contextlevel );
2008-10-30 06:26:18 +00:00
}
2006-08-08 05:13:06 +00:00
2008-11-05 08:17:30 +00:00
/**
2011-10-14 12:48:00 +02:00
* Fix the roles . sortorder field in the database , so it contains sequential integers ,
* and return an array of roleids in order .
2009-05-22 02:03:55 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param array $allroles array of roles , as returned by get_all_roles ();
* @ return array $role -> sortorder =-> $role -> id with the keys in ascending order .
2008-11-05 08:17:30 +00:00
*/
2011-10-14 12:48:00 +02:00
function fix_role_sortorder ( $allroles ) {
global $DB ;
$rolesort = array ();
$i = 0 ;
foreach ( $allroles as $role ) {
$rolesort [ $i ] = $role -> id ;
if ( $role -> sortorder != $i ) {
$r = new stdClass ();
$r -> id = $role -> id ;
$r -> sortorder = $i ;
$DB -> update_record ( 'role' , $r );
$allroles [ $role -> id ] -> sortorder = $i ;
}
$i ++ ;
}
return $rolesort ;
2008-11-05 08:17:30 +00:00
}
/**
2011-10-14 12:48:00 +02:00
* Switch the sort order of two roles ( used in admin / roles / manage . php ) .
2009-05-19 07:50:54 +00:00
*
2012-01-14 19:12:43 +01:00
* @ param stdClass $first The first role . Actually , only -> sortorder is used .
* @ param stdClass $second The second role . Actually , only -> sortorder is used .
2011-10-14 12:48:00 +02:00
* @ return boolean success or failure
2008-11-05 08:17:30 +00:00
*/
2011-10-14 12:48:00 +02:00
function switch_roles ( $first , $second ) {
global $DB ;
$temp = $DB -> get_field ( 'role' , 'MAX(sortorder) + 1' , array ());
$result = $DB -> set_field ( 'role' , 'sortorder' , $temp , array ( 'sortorder' => $first -> sortorder ));
$result = $result && $DB -> set_field ( 'role' , 'sortorder' , $first -> sortorder , array ( 'sortorder' => $second -> sortorder ));
$result = $result && $DB -> set_field ( 'role' , 'sortorder' , $second -> sortorder , array ( 'sortorder' => $temp ));
return $result ;
2008-11-05 08:17:30 +00:00
}
2006-08-08 05:13:06 +00:00
/**
2011-10-14 12:48:00 +02:00
* Duplicates all the base definitions of a role
2009-05-19 07:50:54 +00:00
*
2012-01-14 19:12:43 +01:00
* @ param stdClass $sourcerole role to copy from
2011-10-14 12:48:00 +02:00
* @ param int $targetrole id of role to copy to
2006-08-08 05:13:06 +00:00
*/
2011-10-14 12:48:00 +02:00
function role_cap_duplicate ( $sourcerole , $targetrole ) {
global $DB ;
2006-08-08 05:13:06 +00:00
2011-10-14 12:48:00 +02:00
$systemcontext = context_system :: instance ();
$caps = $DB -> get_records_sql ( " SELECT *
FROM { role_capabilities }
WHERE roleid = ? AND contextid = ? " ,
array ( $sourcerole -> id , $systemcontext -> id ));
// adding capabilities
foreach ( $caps as $cap ) {
unset ( $cap -> id );
$cap -> roleid = $targetrole ;
$DB -> insert_record ( 'role_capabilities' , $cap );
2006-09-12 19:54:33 +00:00
}
2006-09-20 08:30:49 +00:00
}
2006-08-08 05:13:06 +00:00
/**
2011-10-14 12:48:00 +02:00
* Returns two lists , this can be used to find out if user has capability .
* Having any needed role and no forbidden role in this context means
* user has this capability in this context .
* Use get_role_names_with_cap_in_context () if you need role names to display in the UI
2009-05-22 02:03:55 +00:00
*
2012-01-14 19:12:43 +01:00
* @ param stdClass $context
2011-10-14 12:48:00 +02:00
* @ param string $capability
* @ return array ( $neededroles , $forbiddenroles )
2006-08-08 05:13:06 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_roles_with_cap_in_context ( $context , $capability ) {
2008-05-24 18:35:48 +00:00
global $DB ;
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
$ctxids = trim ( $context -> path , '/' ); // kill leading slash
$ctxids = str_replace ( '/' , ',' , $ctxids );
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
$sql = " SELECT rc.id, rc.roleid, rc.permission, ctx.depth
FROM { role_capabilities } rc
JOIN { context } ctx ON ctx . id = rc . contextid
WHERE rc . capability = : cap AND ctx . id IN ( $ctxids )
ORDER BY rc . roleid ASC , ctx . depth DESC " ;
$params = array ( 'cap' => $capability );
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
if ( ! $capdefs = $DB -> get_records_sql ( $sql , $params )) {
// no cap definitions --> no capability
return array ( array (), array ());
2006-08-28 08:42:30 +00:00
}
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
$forbidden = array ();
$needed = array ();
foreach ( $capdefs as $def ) {
if ( isset ( $forbidden [ $def -> roleid ])) {
continue ;
}
if ( $def -> permission == CAP_PROHIBIT ) {
$forbidden [ $def -> roleid ] = $def -> roleid ;
unset ( $needed [ $def -> roleid ]);
continue ;
}
if ( ! isset ( $needed [ $def -> roleid ])) {
if ( $def -> permission == CAP_ALLOW ) {
$needed [ $def -> roleid ] = true ;
} else if ( $def -> permission == CAP_PREVENT ) {
$needed [ $def -> roleid ] = false ;
2006-09-20 21:00:45 +00:00
}
2006-09-05 03:07:56 +00:00
}
2006-08-09 13:14:15 +00:00
}
2011-10-14 12:48:00 +02:00
unset ( $capdefs );
2007-01-11 06:06:11 +00:00
2011-10-14 12:48:00 +02:00
// remove all those roles not allowing
foreach ( $needed as $key => $value ) {
if ( ! $value ) {
unset ( $needed [ $key ]);
} else {
$needed [ $key ] = $key ;
}
2008-11-21 05:46:45 +00:00
}
2006-08-08 05:13:06 +00:00
2011-10-14 12:48:00 +02:00
return array ( $needed , $forbidden );
2006-08-08 05:13:06 +00:00
}
2008-05-12 15:28:34 +00:00
/**
2011-10-14 12:48:00 +02:00
* Returns an array of role IDs that have ALL of the the supplied capabilities
* Uses get_roles_with_cap_in_context () . Returns $allowed minus $forbidden
2008-05-12 15:28:34 +00:00
*
2012-01-14 19:12:43 +01:00
* @ param stdClass $context
2011-10-14 12:48:00 +02:00
* @ param array $capabilities An array of capabilities
* @ return array of roles with all of the required capabilities
2008-05-12 15:28:34 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_roles_with_caps_in_context ( $context , $capabilities ) {
$neededarr = array ();
$forbiddenarr = array ();
foreach ( $capabilities as $caprequired ) {
list ( $neededarr [], $forbiddenarr []) = get_roles_with_cap_in_context ( $context , $caprequired );
2008-05-12 15:28:34 +00:00
}
2006-09-20 08:30:49 +00:00
2011-10-14 12:48:00 +02:00
$rolesthatcanrate = array ();
if ( ! empty ( $neededarr )) {
foreach ( $neededarr as $needed ) {
if ( empty ( $rolesthatcanrate )) {
$rolesthatcanrate = $needed ;
} else {
//only want roles that have all caps
$rolesthatcanrate = array_intersect_key ( $rolesthatcanrate , $needed );
}
}
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
if ( ! empty ( $forbiddenarr ) && ! empty ( $rolesthatcanrate )) {
foreach ( $forbiddenarr as $forbidden ) {
//remove any roles that are forbidden any of the caps
$rolesthatcanrate = array_diff ( $rolesthatcanrate , $forbidden );
}
}
return $rolesthatcanrate ;
2010-03-31 07:41:31 +00:00
}
/**
2011-10-14 12:48:00 +02:00
* Returns an array of role names that have ALL of the the supplied capabilities
* Uses get_roles_with_caps_in_context () . Returns $allowed minus $forbidden
*
2012-01-14 19:12:43 +01:00
* @ param stdClass $context
2011-10-14 12:48:00 +02:00
* @ param array $capabilities An array of capabilities
* @ return array of roles with all of the required capabilities
2010-03-31 07:41:31 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_role_names_with_caps_in_context ( $context , $capabilities ) {
global $DB ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
$rolesthatcanrate = get_roles_with_caps_in_context ( $context , $capabilities );
2012-05-18 11:40:35 +02:00
$allroles = $DB -> get_records ( 'role' , null , 'sortorder DESC' );
2010-03-31 07:41:31 +00:00
2012-05-18 11:40:35 +02:00
$roles = array ();
2011-10-14 12:48:00 +02:00
foreach ( $rolesthatcanrate as $r ) {
2012-05-18 11:40:35 +02:00
$roles [ $r ] = $allroles [ $r ];
2011-10-14 12:48:00 +02:00
}
2012-05-18 11:40:35 +02:00
return role_fix_names ( $roles , $context , ROLENAME_ALIAS , true );
2010-03-31 07:41:31 +00:00
}
2008-11-11 07:23:25 +00:00
/**
2011-10-14 12:48:00 +02:00
* This function verifies the prohibit comes from this context
* and there are no more prohibits in parent contexts .
2009-05-22 02:03:55 +00:00
*
2011-10-14 12:48:00 +02:00
* @ param int $roleid
* @ param context $context
* @ param string $capability name
2009-11-01 11:31:16 +00:00
* @ return bool
2008-11-11 07:23:25 +00:00
*/
2011-10-14 12:48:00 +02:00
function prohibit_is_removable ( $roleid , context $context , $capability ) {
global $DB ;
$ctxids = trim ( $context -> path , '/' ); // kill leading slash
$ctxids = str_replace ( '/' , ',' , $ctxids );
$params = array ( 'roleid' => $roleid , 'cap' => $capability , 'prohibit' => CAP_PROHIBIT );
$sql = " SELECT ctx.id
FROM { role_capabilities } rc
JOIN { context } ctx ON ctx . id = rc . contextid
WHERE rc . roleid = : roleid AND rc . permission = : prohibit AND rc . capability = : cap AND ctx . id IN ( $ctxids )
ORDER BY ctx . depth DESC " ;
if ( ! $prohibits = $DB -> get_records_sql ( $sql , $params )) {
// no prohibits == nothing to remove
return true ;
}
if ( count ( $prohibits ) > 1 ) {
2012-01-14 19:12:43 +01:00
// more prohibits can not be removed
2011-10-14 12:48:00 +02:00
return false ;
}
return ! empty ( $prohibits [ $context -> id ]);
2008-11-11 07:23:25 +00:00
}
2008-11-25 07:29:14 +00:00
/**
2011-10-14 12:48:00 +02:00
* More user friendly role permission changing ,
* it should produce as few overrides as possible .
2012-01-14 19:12:43 +01:00
*
2011-10-14 12:48:00 +02:00
* @ param int $roleid
2012-01-14 19:12:43 +01:00
* @ param stdClass $context
2011-10-14 12:48:00 +02:00
* @ param string $capname capability name
* @ param int $permission
* @ return void
2008-11-25 07:29:14 +00:00
*/
2011-10-14 12:48:00 +02:00
function role_change_permission ( $roleid , $context , $capname , $permission ) {
2008-11-25 07:29:14 +00:00
global $DB ;
2011-10-14 12:48:00 +02:00
if ( $permission == CAP_INHERIT ) {
unassign_capability ( $capname , $roleid , $context -> id );
$context -> mark_dirty ();
return ;
2008-11-25 07:29:14 +00:00
}
2011-10-14 12:48:00 +02:00
$ctxids = trim ( $context -> path , '/' ); // kill leading slash
$ctxids = str_replace ( '/' , ',' , $ctxids );
2007-02-09 02:24:59 +00:00
2011-10-14 12:48:00 +02:00
$params = array ( 'roleid' => $roleid , 'cap' => $capname );
2007-09-19 06:52:34 +00:00
2011-10-14 12:48:00 +02:00
$sql = " SELECT ctx.id, rc.permission, ctx.depth
FROM { role_capabilities } rc
JOIN { context } ctx ON ctx . id = rc . contextid
WHERE rc . roleid = : roleid AND rc . capability = : cap AND ctx . id IN ( $ctxids )
ORDER BY ctx . depth DESC " ;
2007-02-09 02:24:59 +00:00
2011-10-14 12:48:00 +02:00
if ( $existing = $DB -> get_records_sql ( $sql , $params )) {
foreach ( $existing as $e ) {
if ( $e -> permission == CAP_PROHIBIT ) {
// prohibit can not be overridden, no point in changing anything
return ;
2007-02-09 02:24:59 +00:00
}
2011-10-14 12:48:00 +02:00
}
$lowest = array_shift ( $existing );
if ( $lowest -> permission == $permission ) {
// permission already set in this context or parent - nothing to do
return ;
}
if ( $existing ) {
$parent = array_shift ( $existing );
if ( $parent -> permission == $permission ) {
// permission already set in parent context or parent - just unset in this context
// we do this because we want as few overrides as possible for performance reasons
unassign_capability ( $capname , $roleid , $context -> id );
$context -> mark_dirty ();
return ;
2009-07-13 08:37:34 +00:00
}
2011-10-14 12:48:00 +02:00
}
2007-02-09 02:24:59 +00:00
2011-10-14 12:48:00 +02:00
} else {
if ( $permission == CAP_PREVENT ) {
// nothing means role does not have permission
return ;
}
}
2007-02-09 02:24:59 +00:00
2011-10-14 12:48:00 +02:00
// assign the needed capability
assign_capability ( $capname , $permission , $roleid , $context -> id , true );
2007-02-09 02:24:59 +00:00
2011-10-14 12:48:00 +02:00
// force cap reloading
$context -> mark_dirty ();
2007-02-09 02:24:59 +00:00
}
2006-09-20 08:30:49 +00:00
/**
2011-10-14 12:48:00 +02:00
* Basic moodle context abstraction class .
2009-05-19 07:50:54 +00:00
*
2012-02-21 11:48:33 +13:00
* Google confirms that no other important framework is using " context " class ,
* we could use something else like mcontext or moodle_context , but we need to type
* this very often which would be annoying and it would take too much space ...
*
* This class is derived from stdClass for backwards compatibility with
* odl $context record that was returned from DML $DB -> get_record ()
*
2012-01-14 19:12:43 +01:00
* @ package core_access
* @ category access
* @ copyright Petr Skoda { @ link http :// skodak . org }
* @ license http :// www . gnu . org / copyleft / gpl . html GNU GPL v3 or later
* @ since 2.2
2011-10-14 12:48:00 +02:00
*
* @ property - read int $id context id
* @ property - read int $contextlevel CONTEXT_SYSTEM , CONTEXT_COURSE , etc .
* @ property - read int $instanceid id of related instance in each context
* @ property - read string $path path to context , starts with system context
2012-02-21 11:48:33 +13:00
* @ property - read int $depth
2010-03-31 07:41:31 +00:00
*/
2012-03-28 11:38:53 +08:00
abstract class context extends stdClass implements IteratorAggregate {
2010-03-31 07:41:31 +00:00
2012-02-21 11:48:33 +13:00
/**
* The context id
* Can be accessed publicly through $context -> id
* @ var int
2011-10-14 12:48:00 +02:00
*/
protected $_id ;
2012-02-21 11:48:33 +13:00
/**
* The context level
* Can be accessed publicly through $context -> contextlevel
* @ var int One of CONTEXT_ * e . g . CONTEXT_COURSE , CONTEXT_MODULE
*/
2011-10-14 12:48:00 +02:00
protected $_contextlevel ;
2012-02-21 11:48:33 +13:00
/**
* Id of the item this context is related to e . g . COURSE_CONTEXT => course . id
* Can be accessed publicly through $context -> instanceid
* @ var int
*/
2011-10-14 12:48:00 +02:00
protected $_instanceid ;
2012-02-21 11:48:33 +13:00
/**
* The path to the context always starting from the system context
* Can be accessed publicly through $context -> path
* @ var string
*/
2011-10-14 12:48:00 +02:00
protected $_path ;
2010-03-31 07:41:31 +00:00
2012-02-21 11:48:33 +13:00
/**
* The depth of the context in relation to parent contexts
* Can be accessed publicly through $context -> depth
* @ var int
*/
protected $_depth ;
2010-03-31 07:41:31 +00:00
2012-02-21 11:48:33 +13:00
/**
* @ var array Context caching info
*/
2011-10-14 12:48:00 +02:00
private static $cache_contextsbyid = array ();
2012-02-21 11:48:33 +13:00
/**
* @ var array Context caching info
*/
2011-10-14 12:48:00 +02:00
private static $cache_contexts = array ();
2010-03-31 07:41:31 +00:00
2012-02-21 11:48:33 +13:00
/**
* Context count
* Why do we do count contexts ? Because count ( $array ) is horribly slow for large arrays
* @ var int
*/
protected static $cache_count = 0 ;
/**
* @ var array Context caching info
*/
2011-10-14 12:48:00 +02:00
protected static $cache_preloaded = array ();
2012-02-21 11:48:33 +13:00
/**
* @ var context_system The system context once initialised
*/
2011-10-14 12:48:00 +02:00
protected static $systemcontext = null ;
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
/**
* Resets the cache to remove all data .
2011-10-25 14:20:22 +02:00
* @ static
2011-10-14 12:48:00 +02:00
*/
protected static function reset_caches () {
self :: $cache_contextsbyid = array ();
self :: $cache_contexts = array ();
self :: $cache_count = 0 ;
self :: $cache_preloaded = array ();
2007-09-13 13:44:35 +00:00
2011-10-14 12:48:00 +02:00
self :: $systemcontext = null ;
}
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
/**
* Adds a context to the cache . If the cache is full , discards a batch of
* older entries .
*
* @ static
* @ param context $context New context to add
* @ return void
*/
protected static function cache_add ( context $context ) {
if ( isset ( self :: $cache_contextsbyid [ $context -> id ])) {
// already cached, no need to do anything - this is relatively cheap, we do all this because count() is slow
return ;
}
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
if ( self :: $cache_count >= CONTEXT_CACHE_MAX_SIZE ) {
$i = 0 ;
foreach ( self :: $cache_contextsbyid as $ctx ) {
$i ++ ;
if ( $i <= 100 ) {
// we want to keep the first contexts to be loaded on this page, hopefully they will be needed again later
continue ;
}
if ( $i > ( CONTEXT_CACHE_MAX_SIZE / 3 )) {
// we remove oldest third of the contexts to make room for more contexts
break ;
}
unset ( self :: $cache_contextsbyid [ $ctx -> id ]);
unset ( self :: $cache_contexts [ $ctx -> contextlevel ][ $ctx -> instanceid ]);
self :: $cache_count -- ;
}
}
2006-08-14 08:14:02 +00:00
2011-10-14 12:48:00 +02:00
self :: $cache_contexts [ $context -> contextlevel ][ $context -> instanceid ] = $context ;
self :: $cache_contextsbyid [ $context -> id ] = $context ;
self :: $cache_count ++ ;
}
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
/**
* Removes a context from the cache .
*
* @ static
* @ param context $context Context object to remove
* @ return void
*/
protected static function cache_remove ( context $context ) {
if ( ! isset ( self :: $cache_contextsbyid [ $context -> id ])) {
// not cached, no need to do anything - this is relatively cheap, we do all this because count() is slow
return ;
}
unset ( self :: $cache_contexts [ $context -> contextlevel ][ $context -> instanceid ]);
unset ( self :: $cache_contextsbyid [ $context -> id ]);
self :: $cache_count -- ;
if ( self :: $cache_count < 0 ) {
self :: $cache_count = 0 ;
}
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Gets a context from the cache .
*
* @ static
* @ param int $contextlevel Context level
* @ param int $instance Instance ID
* @ return context | bool Context or false if not in cache
*/
protected static function cache_get ( $contextlevel , $instance ) {
if ( isset ( self :: $cache_contexts [ $contextlevel ][ $instance ])) {
return self :: $cache_contexts [ $contextlevel ][ $instance ];
}
return false ;
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Gets a context from the cache based on its id .
*
* @ static
* @ param int $id Context ID
* @ return context | bool Context or false if not in cache
*/
protected static function cache_get_by_id ( $id ) {
if ( isset ( self :: $cache_contextsbyid [ $id ])) {
return self :: $cache_contextsbyid [ $id ];
}
return false ;
2010-03-31 07:41:31 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Preloads context information from db record and strips the cached info .
*
* @ static
* @ param stdClass $rec
* @ return void ( modifies $rec )
*/
protected static function preload_from_record ( stdClass $rec ) {
if ( empty ( $rec -> ctxid ) or empty ( $rec -> ctxlevel ) or empty ( $rec -> ctxinstance ) or empty ( $rec -> ctxpath ) or empty ( $rec -> ctxdepth )) {
// $rec does not have enough data, passed here repeatedly or context does not exist yet
return ;
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
// note: in PHP5 the objects are passed by reference, no need to return $rec
$record = new stdClass ();
$record -> id = $rec -> ctxid ; unset ( $rec -> ctxid );
$record -> contextlevel = $rec -> ctxlevel ; unset ( $rec -> ctxlevel );
$record -> instanceid = $rec -> ctxinstance ; unset ( $rec -> ctxinstance );
$record -> path = $rec -> ctxpath ; unset ( $rec -> ctxpath );
$record -> depth = $rec -> ctxdepth ; unset ( $rec -> ctxdepth );
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
return context :: create_instance_from_record ( $record );
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
// ====== magic methods =======
2008-02-29 08:44:23 +00:00
2011-10-14 12:48:00 +02:00
/**
* Magic setter method , we do not want anybody to modify properties from the outside
* @ param string $name
2012-02-21 11:48:33 +13:00
* @ param mixed $value
2011-10-14 12:48:00 +02:00
*/
public function __set ( $name , $value ) {
debugging ( 'Can not change context instance properties!' );
}
2008-02-29 08:44:23 +00:00
2011-10-14 12:48:00 +02:00
/**
* Magic method getter , redirects to read only values .
* @ param string $name
* @ return mixed
*/
public function __get ( $name ) {
switch ( $name ) {
case 'id' : return $this -> _id ;
case 'contextlevel' : return $this -> _contextlevel ;
case 'instanceid' : return $this -> _instanceid ;
case 'path' : return $this -> _path ;
case 'depth' : return $this -> _depth ;
default :
debugging ( 'Invalid context property accessed! ' . $name );
return null ;
2008-02-29 08:44:23 +00:00
}
2006-08-15 08:29:29 +00:00
}
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
/**
* Full support for isset on our magic read only properties .
2012-02-21 11:48:33 +13:00
* @ param string $name
2011-10-14 12:48:00 +02:00
* @ return bool
*/
public function __isset ( $name ) {
switch ( $name ) {
case 'id' : return isset ( $this -> _id );
case 'contextlevel' : return isset ( $this -> _contextlevel );
case 'instanceid' : return isset ( $this -> _instanceid );
case 'path' : return isset ( $this -> _path );
case 'depth' : return isset ( $this -> _depth );
2006-08-18 08:01:16 +00:00
2011-10-14 12:48:00 +02:00
default : return false ;
}
2006-09-20 21:00:45 +00:00
2006-08-18 08:01:16 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* ALl properties are read only , sorry .
* @ param string $name
*/
public function __unset ( $name ) {
debugging ( 'Can not unset context instance properties!' );
2006-08-18 08:01:16 +00:00
}
2006-09-20 21:00:45 +00:00
2012-03-28 11:38:53 +08:00
// ====== implementing method from interface IteratorAggregate ======
/**
* Create an iterator because magic vars can 't be seen by ' foreach ' .
*
* Now we can convert context object to array using convert_to_array (),
* and feed it properly to json_encode () .
*/
public function getIterator () {
$ret = array (
'id' => $this -> id ,
'contextlevel' => $this -> contextlevel ,
'instanceid' => $this -> instanceid ,
'path' => $this -> path ,
'depth' => $this -> depth
);
return new ArrayIterator ( $ret );
}
2011-10-14 12:48:00 +02:00
// ====== general context methods ======
2006-08-18 08:01:16 +00:00
2011-10-14 12:48:00 +02:00
/**
* Constructor is protected so that devs are forced to
* use context_xxx :: instance () or context :: instance_by_id () .
*
* @ param stdClass $record
*/
protected function __construct ( stdClass $record ) {
$this -> _id = $record -> id ;
$this -> _contextlevel = ( int ) $record -> contextlevel ;
$this -> _instanceid = $record -> instanceid ;
$this -> _path = $record -> path ;
$this -> _depth = $record -> depth ;
}
2010-03-31 07:41:31 +00:00
2011-10-14 12:48:00 +02:00
/**
* This function is also used to work around 'protected' keyword problems in context_helper .
2011-10-25 14:20:22 +02:00
* @ static
2011-10-14 12:48:00 +02:00
* @ param stdClass $record
* @ return context instance
*/
protected static function create_instance_from_record ( stdClass $record ) {
$classname = context_helper :: get_class_for_level ( $record -> contextlevel );
2006-09-23 12:51:00 +00:00
2011-10-14 12:48:00 +02:00
if ( $context = context :: cache_get_by_id ( $record -> id )) {
return $context ;
}
2006-08-19 08:12:45 +00:00
2011-10-14 12:48:00 +02:00
$context = new $classname ( $record );
context :: cache_add ( $context );
return $context ;
}
/**
* Copy prepared new contexts from temp table to context table ,
* we do this in db specific way for perf reasons only .
* @ static
*/
protected static function merge_context_temp_table () {
global $DB ;
/* MDL - 11347 :
* - mysql does not allow to use FROM in UPDATE statements
* - using two tables after UPDATE works in mysql , but might give unexpected
* results in pg 8 ( depends on configuration )
* - using table alias in UPDATE does not work in pg < 8.2
*
* Different code for each database - mostly for performance reasons
*/
$dbfamily = $DB -> get_dbfamily ();
if ( $dbfamily == 'mysql' ) {
$updatesql = " UPDATE { context} ct, { context_temp} temp
SET ct . path = temp . path ,
ct . depth = temp . depth
WHERE ct . id = temp . id " ;
} else if ( $dbfamily == 'oracle' ) {
$updatesql = " UPDATE { context} ct
SET ( ct . path , ct . depth ) =
( SELECT temp . path , temp . depth
FROM { context_temp } temp
WHERE temp . id = ct . id )
WHERE EXISTS ( SELECT 'x'
FROM { context_temp } temp
WHERE temp . id = ct . id ) " ;
} else if ( $dbfamily == 'postgres' or $dbfamily == 'mssql' ) {
$updatesql = " UPDATE { context}
SET path = temp . path ,
depth = temp . depth
FROM { context_temp } temp
WHERE temp . id = { context } . id " ;
} else {
// sqlite and others
$updatesql = " UPDATE { context}
SET path = ( SELECT path FROM { context_temp } WHERE id = { context } . id ),
depth = ( SELECT depth FROM { context_temp } WHERE id = { context } . id )
WHERE id IN ( SELECT id FROM { context_temp }) " ;
2006-08-19 08:12:45 +00:00
}
2011-10-14 12:48:00 +02:00
$DB -> execute ( $updatesql );
2006-08-19 08:12:45 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Get a context instance as an object , from a given context id .
*
* @ static
* @ param int $id context id
* @ param int $strictness IGNORE_MISSING means compatible mode , false returned if record not found , debug message if more found ;
* MUST_EXIST means throw exception if no record found
* @ return context | bool the context object or false if not found
*/
public static function instance_by_id ( $id , $strictness = MUST_EXIST ) {
global $DB ;
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
if ( get_called_class () !== 'context' and get_called_class () !== 'context_helper' ) {
// some devs might confuse context->id and instanceid, better prevent these mistakes completely
throw new coding_exception ( 'use only context::instance_by_id() for real context levels use ::instance() methods' );
}
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
if ( $id == SYSCONTEXTID ) {
return context_system :: instance ( 0 , $strictness );
}
2008-02-05 12:59:28 +00:00
2011-10-14 12:48:00 +02:00
if ( is_array ( $id ) or is_object ( $id ) or empty ( $id )) {
throw new coding_exception ( 'Invalid context id specified context::instance_by_id()' );
}
2006-08-18 08:01:16 +00:00
2011-10-14 12:48:00 +02:00
if ( $context = context :: cache_get_by_id ( $id )) {
return $context ;
}
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
if ( $record = $DB -> get_record ( 'context' , array ( 'id' => $id ), '*' , $strictness )) {
return context :: create_instance_from_record ( $record );
}
2006-08-23 06:36:08 +00:00
2011-10-14 12:48:00 +02:00
return false ;
}
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
/**
* Update context info after moving context in the tree structure .
*
* @ param context $newparent
* @ return void
*/
public function update_moved ( context $newparent ) {
global $DB ;
2009-03-25 04:20:57 +00:00
2011-10-14 12:48:00 +02:00
$frompath = $this -> _path ;
$newpath = $newparent -> path . '/' . $this -> _id ;
2009-03-25 04:20:57 +00:00
2011-10-14 12:48:00 +02:00
$trans = $DB -> start_delegated_transaction ();
2006-08-23 06:36:08 +00:00
2011-10-14 12:48:00 +02:00
$this -> mark_dirty ();
2007-09-19 07:27:46 +00:00
2011-10-14 12:48:00 +02:00
$setdepth = '' ;
if (( $newparent -> depth + 1 ) != $this -> _depth ) {
$diff = $newparent -> depth - $this -> _depth + 1 ;
$setdepth = " , depth = depth + $diff " ;
}
$sql = " UPDATE { context}
SET path = ?
$setdepth
WHERE id = ? " ;
$params = array ( $newpath , $this -> _id );
$DB -> execute ( $sql , $params );
$this -> _path = $newpath ;
$this -> _depth = $newparent -> depth + 1 ;
$sql = " UPDATE { context}
SET path = " . $DB->sql_concat ( " ? " , $DB->sql_substr ( " path " , strlen( $frompath )+1)). "
$setdepth
WHERE path LIKE ? " ;
$params = array ( $newpath , " { $frompath } /% " );
$DB -> execute ( $sql , $params );
$this -> mark_dirty ();
context :: reset_caches ();
$trans -> allow_commit ();
2010-07-19 17:51:17 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Remove all context path info and optionally rebuild it .
*
* @ param bool $rebuild
* @ return void
*/
public function reset_paths ( $rebuild = true ) {
global $DB ;
if ( $this -> _path ) {
$this -> mark_dirty ();
}
$DB -> set_field_select ( 'context' , 'depth' , 0 , " path LIKE '%/ $this->_id /%' " );
$DB -> set_field_select ( 'context' , 'path' , NULL , " path LIKE '%/ $this->_id /%' " );
if ( $this -> _contextlevel != CONTEXT_SYSTEM ) {
$DB -> set_field ( 'context' , 'depth' , 0 , array ( 'id' => $this -> _id ));
$DB -> set_field ( 'context' , 'path' , NULL , array ( 'id' => $this -> _id ));
$this -> _depth = 0 ;
$this -> _path = null ;
2008-11-06 07:34:01 +00:00
}
2008-07-23 16:10:06 +00:00
2011-10-14 12:48:00 +02:00
if ( $rebuild ) {
context_helper :: build_all_paths ( false );
}
2008-07-23 16:10:06 +00:00
2011-10-14 12:48:00 +02:00
context :: reset_caches ();
2008-10-30 06:26:18 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Delete all data linked to content , do not delete the context record itself
*/
public function delete_content () {
global $CFG , $DB ;
2008-10-30 06:26:18 +00:00
2011-10-14 12:48:00 +02:00
blocks_delete_all_for_context ( $this -> _id );
filter_delete_all_for_context ( $this -> _id );
require_once ( $CFG -> dirroot . '/comment/lib.php' );
comment :: delete_comments ( array ( 'contextid' => $this -> _id ));
require_once ( $CFG -> dirroot . '/rating/lib.php' );
$delopt = new stdclass ();
$delopt -> contextid = $this -> _id ;
$rm = new rating_manager ();
$rm -> delete_ratings ( $delopt );
// delete all files attached to this context
$fs = get_file_storage ();
$fs -> delete_area_files ( $this -> _id );
2011-11-13 13:50:00 +01:00
// delete all advanced grading data attached to this context
require_once ( $CFG -> dirroot . '/grade/grading/lib.php' );
grading_manager :: delete_all_for_context ( $this -> _id );
2011-10-14 12:48:00 +02:00
// now delete stuff from role related tables, role_unassign_all
// and unenrol should be called earlier to do proper cleanup
$DB -> delete_records ( 'role_assignments' , array ( 'contextid' => $this -> _id ));
$DB -> delete_records ( 'role_capabilities' , array ( 'contextid' => $this -> _id ));
$DB -> delete_records ( 'role_names' , array ( 'contextid' => $this -> _id ));
2009-01-17 23:16:20 +00:00
}
2007-09-19 07:27:46 +00:00
2011-10-14 12:48:00 +02:00
/**
* Delete the context content and the context record itself
*/
public function delete () {
global $DB ;
// double check the context still exists
if ( ! $DB -> record_exists ( 'context' , array ( 'id' => $this -> _id ))) {
context :: cache_remove ( $this );
return ;
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
}
2011-10-14 12:48:00 +02:00
$this -> delete_content ();
$DB -> delete_records ( 'context' , array ( 'id' => $this -> _id ));
// purge static context cache if entry present
context :: cache_remove ( $this );
// do not mark dirty contexts if parents unknown
if ( ! is_null ( $this -> _path ) and $this -> _depth > 0 ) {
$this -> mark_dirty ();
2010-03-31 07:41:31 +00:00
}
2008-10-30 06:26:18 +00:00
}
2008-04-18 06:21:00 +00:00
2011-10-14 12:48:00 +02:00
// ====== context level related methods ======
/**
* Utility method for context creation
*
* @ static
* @ param int $contextlevel
* @ param int $instanceid
* @ param string $parentpath
* @ return stdClass context record
*/
protected static function insert_context_record ( $contextlevel , $instanceid , $parentpath ) {
global $DB ;
$record = new stdClass ();
$record -> contextlevel = $contextlevel ;
$record -> instanceid = $instanceid ;
$record -> depth = 0 ;
$record -> path = null ; //not known before insert
$record -> id = $DB -> insert_record ( 'context' , $record );
// now add path if known - it can be added later
if ( ! is_null ( $parentpath )) {
$record -> path = $parentpath . '/' . $record -> id ;
$record -> depth = substr_count ( $record -> path , '/' );
$DB -> update_record ( 'context' , $record );
}
return $record ;
2008-10-30 06:26:18 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns human readable context identifier .
*
* @ param boolean $withprefix whether to prefix the name of the context with the
* type of context , e . g . User , Course , Forum , etc .
* @ param boolean $short whether to use the short name of the thing . Only applies
* to course contexts
* @ return string the human readable context name .
*/
public function get_context_name ( $withprefix = true , $short = false ) {
// must be implemented in all context levels
throw new coding_exception ( 'can not get name of abstract context' );
}
2008-02-15 16:40:45 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns the most relevant URL for this context .
*
* @ return moodle_url
*/
public abstract function get_url ();
2008-02-15 16:40:45 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns array of relevant context capability records .
*
* @ return array
*/
public abstract function get_capabilities ();
2009-03-23 08:15:21 +00:00
2011-10-14 12:48:00 +02:00
/**
* Recursive function which , given a context , find all its children context ids .
*
* For course category contexts it will return immediate children and all subcategory contexts .
* It will NOT recurse into courses or subcategories categories .
* If you want to do that , call it on the returned courses / categories .
*
* When called for a course context , it will return the modules and blocks
* displayed in the course page and blocks displayed on the module pages .
*
* If called on a user / course / module context it _will_ populate the cache with the appropriate
* contexts ; - )
*
* @ return array Array of child records
*/
public function get_child_contexts () {
global $DB ;
2009-03-23 08:15:21 +00:00
2011-10-14 12:48:00 +02:00
$sql = " SELECT ctx.*
FROM { context } ctx
WHERE ctx . path LIKE ? " ;
$params = array ( $this -> _path . '/%' );
$records = $DB -> get_records_sql ( $sql , $params );
2009-03-23 08:15:21 +00:00
2011-10-14 12:48:00 +02:00
$result = array ();
foreach ( $records as $record ) {
$result [ $record -> id ] = context :: create_instance_from_record ( $record );
}
2008-02-15 16:40:45 +00:00
2011-10-14 12:48:00 +02:00
return $result ;
}
2006-08-23 06:36:08 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns parent contexts of this context in reversed order , i . e . parent first ,
* then grand parent , etc .
*
* @ param bool $includeself tre means include self too
* @ return array of context instances
*/
public function get_parent_contexts ( $includeself = false ) {
if ( ! $contextids = $this -> get_parent_context_ids ( $includeself )) {
2008-11-06 07:34:01 +00:00
return array ();
}
2008-07-23 16:10:06 +00:00
2011-10-14 12:48:00 +02:00
$result = array ();
foreach ( $contextids as $contextid ) {
$parent = context :: instance_by_id ( $contextid , MUST_EXIST );
$result [ $parent -> id ] = $parent ;
}
2008-07-23 16:10:06 +00:00
2011-10-14 12:48:00 +02:00
return $result ;
2008-11-05 08:17:30 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns parent contexts of this context in reversed order , i . e . parent first ,
* then grand parent , etc .
*
* @ param bool $includeself tre means include self too
* @ return array of context ids
*/
public function get_parent_context_ids ( $includeself = false ) {
if ( empty ( $this -> _path )) {
return array ();
}
$parentcontexts = trim ( $this -> _path , '/' ); // kill leading slash
$parentcontexts = explode ( '/' , $parentcontexts );
if ( ! $includeself ) {
array_pop ( $parentcontexts ); // and remove its own id
}
return array_reverse ( $parentcontexts );
2008-11-05 08:17:30 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns parent context
*
* @ return context
*/
public function get_parent_context () {
if ( empty ( $this -> _path ) or $this -> _id == SYSCONTEXTID ) {
return false ;
}
2009-11-01 11:31:16 +00:00
2011-10-14 12:48:00 +02:00
$parentcontexts = trim ( $this -> _path , '/' ); // kill leading slash
$parentcontexts = explode ( '/' , $parentcontexts );
array_pop ( $parentcontexts ); // self
$contextid = array_pop ( $parentcontexts ); // immediate parent
return context :: instance_by_id ( $contextid , MUST_EXIST );
2008-07-23 16:10:06 +00:00
}
2006-09-17 07:00:47 +00:00
2011-10-14 12:48:00 +02:00
/**
* Is this context part of any course ? If yes return course context .
*
* @ param bool $strict true means throw exception if not found , false means return false if not found
* @ return course_context context of the enclosing course , null if not found or exception
*/
public function get_course_context ( $strict = true ) {
if ( $strict ) {
throw new coding_exception ( 'Context does not belong to any course.' );
} else {
return false ;
2008-11-05 08:17:30 +00:00
}
}
2011-10-14 12:48:00 +02:00
/**
* Returns sql necessary for purging of stale context instances .
*
* @ static
* @ return string cleanup SQL
*/
protected static function get_cleanup_sql () {
throw new coding_exception ( 'get_cleanup_sql() method must be implemented in all context levels' );
2006-09-17 07:00:47 +00:00
}
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
/**
* Rebuild context paths and depths at context level .
*
* @ static
2012-02-21 11:48:33 +13:00
* @ param bool $force
2011-10-14 12:48:00 +02:00
* @ return void
*/
protected static function build_paths ( $force ) {
throw new coding_exception ( 'build_paths() method must be implemented in all context levels' );
}
2008-11-05 08:17:30 +00:00
2011-10-14 12:48:00 +02:00
/**
* Create missing context instances at given level
*
* @ static
* @ return void
*/
protected static function create_level_instances () {
throw new coding_exception ( 'create_level_instances() method must be implemented in all context levels' );
2008-11-05 08:17:30 +00:00
}
2006-08-28 08:42:30 +00:00
2011-10-14 12:48:00 +02:00
/**
* Reset all cached permissions and definitions if the necessary .
* @ return void
*/
public function reload_if_dirty () {
global $ACCESSLIB_PRIVATE , $USER ;
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
2011-10-14 12:48:00 +02:00
// Load dirty contexts list if needed
if ( CLI_SCRIPT ) {
if ( ! isset ( $ACCESSLIB_PRIVATE -> dirtycontexts )) {
// we do not load dirty flags in CLI and cron
$ACCESSLIB_PRIVATE -> dirtycontexts = array ();
}
} else {
if ( ! isset ( $ACCESSLIB_PRIVATE -> dirtycontexts )) {
if ( ! isset ( $USER -> access [ 'time' ])) {
// nothing was loaded yet, we do not need to check dirty contexts now
return ;
}
// no idea why -2 is there, server cluster time difference maybe... (skodak)
$ACCESSLIB_PRIVATE -> dirtycontexts = get_cache_flags ( 'accesslib/dirtycontexts' , $USER -> access [ 'time' ] - 2 );
}
}
foreach ( $ACCESSLIB_PRIVATE -> dirtycontexts as $path => $unused ) {
if ( $path === $this -> _path or strpos ( $this -> _path , $path . '/' ) === 0 ) {
// reload all capabilities of USER and others - preserving loginas, roleswitches, etc
// and then cleanup any marks of dirtyness... at least from our short term memory! :-)
reload_all_capabilities ();
break ;
}
}
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Mark a context as dirty ( with timestamp ) so as to force reloading of the context .
*/
public function mark_dirty () {
global $CFG , $USER , $ACCESSLIB_PRIVATE ;
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
2011-10-14 12:48:00 +02:00
if ( during_initial_install ()) {
return ;
}
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
2011-10-14 12:48:00 +02:00
// only if it is a non-empty string
if ( is_string ( $this -> _path ) && $this -> _path !== '' ) {
set_cache_flag ( 'accesslib/dirtycontexts' , $this -> _path , 1 , time () + $CFG -> sessiontimeout );
if ( isset ( $ACCESSLIB_PRIVATE -> dirtycontexts )) {
$ACCESSLIB_PRIVATE -> dirtycontexts [ $this -> _path ] = 1 ;
} else {
if ( CLI_SCRIPT ) {
$ACCESSLIB_PRIVATE -> dirtycontexts = array ( $this -> _path => 1 );
} else {
if ( isset ( $USER -> access [ 'time' ])) {
$ACCESSLIB_PRIVATE -> dirtycontexts = get_cache_flags ( 'accesslib/dirtycontexts' , $USER -> access [ 'time' ] - 2 );
} else {
$ACCESSLIB_PRIVATE -> dirtycontexts = array ( $this -> _path => 1 );
}
// flags not loaded yet, it will be done later in $context->reload_if_dirty()
}
}
}
}
2008-11-06 07:34:01 +00:00
}
2008-11-13 08:11:10 +00:00
2011-10-23 12:50:56 +02:00
/**
* Context maintenance and helper methods .
*
* This is " extends context " is a bloody hack that tires to work around the deficiencies
* in the " protected " keyword in PHP , this helps us to hide all the internals of context
* level implementation from the rest of code , the code completion returns what developers need .
*
* Thank you Tim Hunt for helping me with this nasty trick .
*
2012-01-14 19:12:43 +01:00
* @ package core_access
* @ category access
* @ copyright Petr Skoda { @ link http :// skodak . org }
* @ license http :// www . gnu . org / copyleft / gpl . html GNU GPL v3 or later
* @ since 2.2
2011-10-23 12:50:56 +02:00
*/
class context_helper extends context {
2012-02-21 11:48:33 +13:00
/**
* @ var array An array mapping context levels to classes
*/
2011-10-23 12:50:56 +02:00
private static $alllevels = array (
CONTEXT_SYSTEM => 'context_system' ,
CONTEXT_USER => 'context_user' ,
CONTEXT_COURSECAT => 'context_coursecat' ,
CONTEXT_COURSE => 'context_course' ,
CONTEXT_MODULE => 'context_module' ,
CONTEXT_BLOCK => 'context_block' ,
);
/**
* Instance does not make sense here , only static use
*/
protected function __construct () {
}
/**
* Returns a class name of the context level class
*
* @ static
* @ param int $contextlevel ( CONTEXT_SYSTEM , etc . )
* @ return string class name of the context class
*/
public static function get_class_for_level ( $contextlevel ) {
if ( isset ( self :: $alllevels [ $contextlevel ])) {
return self :: $alllevels [ $contextlevel ];
} else {
throw new coding_exception ( 'Invalid context level specified' );
}
}
/**
* Returns a list of all context levels
*
* @ static
* @ return array int => string ( level => level class name )
*/
public static function get_all_levels () {
return self :: $alllevels ;
}
/**
* Remove stale contexts that belonged to deleted instances .
* Ideally all code should cleanup contexts properly , unfortunately accidents happen ...
*
* @ static
* @ return void
*/
public static function cleanup_instances () {
global $DB ;
$sqls = array ();
foreach ( self :: $alllevels as $level => $classname ) {
$sqls [] = $classname :: get_cleanup_sql ();
}
$sql = implode ( " UNION " , $sqls );
// it is probably better to use transactions, it might be faster too
$transaction = $DB -> start_delegated_transaction ();
$rs = $DB -> get_recordset_sql ( $sql );
foreach ( $rs as $record ) {
$context = context :: create_instance_from_record ( $record );
$context -> delete ();
}
$rs -> close ();
$transaction -> allow_commit ();
}
/**
* Create all context instances at the given level and above .
*
* @ static
* @ param int $contextlevel null means all levels
* @ param bool $buildpaths
* @ return void
*/
public static function create_instances ( $contextlevel = null , $buildpaths = true ) {
foreach ( self :: $alllevels as $level => $classname ) {
if ( $contextlevel and $level > $contextlevel ) {
// skip potential sub-contexts
continue ;
}
$classname :: create_level_instances ();
if ( $buildpaths ) {
$classname :: build_paths ( false );
}
}
}
/**
* Rebuild paths and depths in all context levels .
*
* @ static
* @ param bool $force false means add missing only
* @ return void
*/
public static function build_all_paths ( $force = false ) {
foreach ( self :: $alllevels as $classname ) {
$classname :: build_paths ( $force );
}
// reset static course cache - it might have incorrect cached data
accesslib_clear_all_caches ( true );
}
/**
* Resets the cache to remove all data .
* @ static
*/
public static function reset_caches () {
context :: reset_caches ();
}
/**
* Returns all fields necessary for context preloading from user $rec .
*
* This helps with performance when dealing with hundreds of contexts .
*
* @ static
* @ param string $tablealias context table alias in the query
* @ return array ( table . column => alias , ... )
*/
public static function get_preload_record_columns ( $tablealias ) {
return array ( " $tablealias .id " => " ctxid " , " $tablealias .path " => " ctxpath " , " $tablealias .depth " => " ctxdepth " , " $tablealias .contextlevel " => " ctxlevel " , " $tablealias .instanceid " => " ctxinstance " );
}
/**
* Returns all fields necessary for context preloading from user $rec .
*
* This helps with performance when dealing with hundreds of contexts .
*
* @ static
* @ param string $tablealias context table alias in the query
* @ return string
*/
public static function get_preload_record_columns_sql ( $tablealias ) {
return " $tablealias .id AS ctxid, $tablealias .path AS ctxpath, $tablealias .depth AS ctxdepth, $tablealias .contextlevel AS ctxlevel, $tablealias .instanceid AS ctxinstance " ;
}
/**
* Preloads context information from db record and strips the cached info .
*
* The db request has to contain all columns from context_helper :: get_preload_record_columns () .
*
* @ static
* @ param stdClass $rec
* @ return void ( modifies $rec )
*/
public static function preload_from_record ( stdClass $rec ) {
context :: preload_from_record ( $rec );
}
/**
* Preload all contexts instances from course .
*
* To be used if you expect multiple queries for course activities ...
*
* @ static
2012-02-21 11:48:33 +13:00
* @ param int $courseid
2011-10-23 12:50:56 +02:00
*/
public static function preload_course ( $courseid ) {
// Users can call this multiple times without doing any harm
if ( isset ( context :: $cache_preloaded [ $courseid ])) {
return ;
}
$coursecontext = context_course :: instance ( $courseid );
$coursecontext -> get_child_contexts ();
context :: $cache_preloaded [ $courseid ] = true ;
}
/**
* Delete context instance
*
* @ static
* @ param int $contextlevel
* @ param int $instanceid
* @ return void
*/
public static function delete_instance ( $contextlevel , $instanceid ) {
global $DB ;
// double check the context still exists
if ( $record = $DB -> get_record ( 'context' , array ( 'contextlevel' => $contextlevel , 'instanceid' => $instanceid ))) {
$context = context :: create_instance_from_record ( $record );
$context -> delete ();
} else {
// we should try to purge the cache anyway
}
}
/**
* Returns the name of specified context level
*
* @ static
* @ param int $contextlevel
* @ return string name of the context level
*/
public static function get_level_name ( $contextlevel ) {
$classname = context_helper :: get_class_for_level ( $contextlevel );
return $classname :: get_level_name ();
}
/**
* not used
*/
public function get_url () {
}
/**
* not used
*/
public function get_capabilities () {
}
}
2008-11-06 07:34:01 +00:00
/**
2012-01-14 19:12:43 +01:00
* System context class
*
* @ package core_access
* @ category access
* @ copyright Petr Skoda { @ link http :// skodak . org }
* @ license http :// www . gnu . org / copyleft / gpl . html GNU GPL v3 or later
* @ since 2.2
2008-11-06 07:34:01 +00:00
*/
2011-10-14 12:48:00 +02:00
class context_system extends context {
/**
* Please use context_system :: instance () if you need the instance of context .
*
* @ param stdClass $record
*/
protected function __construct ( stdClass $record ) {
parent :: __construct ( $record );
if ( $record -> contextlevel != CONTEXT_SYSTEM ) {
throw new coding_exception ( 'Invalid $record->contextlevel in context_system constructor.' );
}
}
MDL-21782 reworked enrolment framework, the core infrastructure is in place, the basic plugins are all implemented; see the tracker issue for list of unfinished bits, expect more changes and improvements during the next week
AMOS START
MOV [sendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage,enrol_self]
MOV [configsendcoursewelcomemessage,core_admin],[sendcoursewelcomemessage_desc,enrol_self]
MOV [enrolstartdate,core],[enrolstartdate,enrol_self]
MOV [enrolenddate,core],[enrolenddate,enrol_self]
CPY [welcometocourse,core],[welcometocourse,enrol_self]
CPY [welcometocoursetext,core],[welcometocoursetext,enrol_self]
MOV [notenrollable,core],[notenrollable,core_enrol]
MOV [enrolenddaterror,core],[enrolenddaterror,enrol_self]
MOV [enrolmentkeyhint,core],[passwordinvalidhint,enrol_self]
MOV [coursemanager,core_admin],[coursecontact,core_admin]
MOV [configcoursemanager,core_admin],[coursecontact_desc,core_admin]
MOV [enrolledincourserole,core],[enrolledincourserole,enrol_manual]
MOV [enrolme,core],[enrolme,core_enrol]
MOV [unenrol,core],[unenrol,core_enrol]
MOV [unenrolme,core],[unenrolme,core_enrol]
MOV [enrolmentnew,core],[enrolmentnew,core_enrol]
MOV [enrolmentnewuser,core],[enrolmentnewuser,core_enrol]
MOV [enrolments,core],[enrolments,core_enrol]
MOV [enrolperiod,core],[enrolperiod,core_enrol]
MOV [unenrolroleusers,core],[unenrolroleusers,core_enrol]
AMOS END
2010-06-21 15:30:49 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns human readable context level name .
*
* @ static
* @ return string the human readable context level name .
*/
2012-01-08 15:39:03 +01:00
public static function get_level_name () {
2011-10-14 12:48:00 +02:00
return get_string ( 'coresystem' );
2008-11-18 07:10:00 +00:00
}
2008-11-06 07:34:01 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns human readable context identifier .
*
* @ param boolean $withprefix does not apply to system context
* @ param boolean $short does not apply to system context
* @ return string the human readable context name .
*/
public function get_context_name ( $withprefix = true , $short = false ) {
return self :: get_level_name ();
2008-11-06 07:34:01 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns the most relevant URL for this context .
*
* @ return moodle_url
*/
public function get_url () {
return new moodle_url ( '/' );
}
/**
* Returns array of relevant context capability records .
*
* @ return array
*/
public function get_capabilities () {
global $DB ;
$sort = 'ORDER BY contextlevel,component,name' ; // To group them sensibly for display
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
$params = array ();
$sql = " SELECT *
FROM { capabilities } " ;
return $DB -> get_records_sql ( $sql . ' ' . $sort , $params );
}
/**
* Create missing context instances at system context
* @ static
*/
protected static function create_level_instances () {
// nothing to do here, the system context is created automatically in installer
self :: instance ( 0 );
2010-03-07 09:28:54 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns system context instance .
*
* @ static
* @ param int $instanceid
* @ param int $strictness
* @ param bool $cache
* @ return context_system context instance
*/
public static function instance ( $instanceid = 0 , $strictness = MUST_EXIST , $cache = true ) {
global $DB ;
if ( $instanceid != 0 ) {
debugging ( 'context_system::instance(): invalid $id parameter detected, should be 0' );
}
if ( defined ( 'SYSCONTEXTID' ) and $cache ) { // dangerous: define this in config.php to eliminate 1 query/page
if ( ! isset ( context :: $systemcontext )) {
$record = new stdClass ();
$record -> id = SYSCONTEXTID ;
$record -> contextlevel = CONTEXT_SYSTEM ;
$record -> instanceid = 0 ;
$record -> path = '/' . SYSCONTEXTID ;
$record -> depth = 1 ;
context :: $systemcontext = new context_system ( $record );
}
return context :: $systemcontext ;
}
try {
// we ignore the strictness completely because system context must except except during install
$record = $DB -> get_record ( 'context' , array ( 'contextlevel' => CONTEXT_SYSTEM ), '*' , MUST_EXIST );
} catch ( dml_exception $e ) {
//table or record does not exist
if ( ! during_initial_install ()) {
// do not mess with system context after install, it simply must exist
throw $e ;
}
$record = null ;
}
if ( ! $record ) {
$record = new stdClass ();
$record -> contextlevel = CONTEXT_SYSTEM ;
$record -> instanceid = 0 ;
$record -> depth = 1 ;
$record -> path = null ; //not known before insert
try {
if ( $DB -> count_records ( 'context' )) {
// contexts already exist, this is very weird, system must be first!!!
return null ;
}
if ( defined ( 'SYSCONTEXTID' )) {
// this would happen only in unittest on sites that went through weird 1.7 upgrade
$record -> id = SYSCONTEXTID ;
$DB -> import_record ( 'context' , $record );
$DB -> get_manager () -> reset_sequence ( 'context' );
} else {
$record -> id = $DB -> insert_record ( 'context' , $record );
}
} catch ( dml_exception $e ) {
// can not create context - table does not exist yet, sorry
return null ;
}
}
if ( $record -> instanceid != 0 ) {
// this is very weird, somebody must be messing with context table
debugging ( 'Invalid system context detected' );
}
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
if ( $record -> depth != 1 or $record -> path != '/' . $record -> id ) {
// fix path if necessary, initial install or path reset
$record -> depth = 1 ;
$record -> path = '/' . $record -> id ;
$DB -> update_record ( 'context' , $record );
}
2008-01-06 23:22:26 +00:00
2011-10-14 12:48:00 +02:00
if ( ! defined ( 'SYSCONTEXTID' )) {
define ( 'SYSCONTEXTID' , $record -> id );
2008-01-06 23:22:26 +00:00
}
2011-10-14 12:48:00 +02:00
context :: $systemcontext = new context_system ( $record );
return context :: $systemcontext ;
2008-01-06 23:22:26 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns all site contexts except the system context , DO NOT call on production servers !!
*
* Contexts are not cached .
*
* @ return array
*/
public function get_child_contexts () {
global $DB ;
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
debugging ( 'Fetching of system context child courses is strongly discouraged on production servers (it may eat all available memory)!' );
2007-09-27 01:46:41 +00:00
2011-10-14 12:48:00 +02:00
// Just get all the contexts except for CONTEXT_SYSTEM level
// and hope we don't OOM in the process - don't cache
$sql = " SELECT c.*
FROM { context } c
WHERE contextlevel > " .CONTEXT_SYSTEM;
$records = $DB -> get_records_sql ( $sql );
2008-01-06 23:22:26 +00:00
2011-10-14 12:48:00 +02:00
$result = array ();
foreach ( $records as $record ) {
$result [ $record -> id ] = context :: create_instance_from_record ( $record );
}
return $result ;
2008-01-06 23:23:58 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns sql necessary for purging of stale context instances .
*
* @ static
* @ return string cleanup SQL
*/
protected static function get_cleanup_sql () {
$sql = "
SELECT c .*
FROM { context } c
WHERE 1 = 2
" ;
return $sql ;
2009-10-05 17:08:16 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Rebuild context paths and depths at system context level .
*
* @ static
2012-02-21 11:48:33 +13:00
* @ param bool $force
2011-10-14 12:48:00 +02:00
*/
protected static function build_paths ( $force ) {
global $DB ;
/* note: ignore $force here, we always do full test of system context */
// exactly one record must exist
$record = $DB -> get_record ( 'context' , array ( 'contextlevel' => CONTEXT_SYSTEM ), '*' , MUST_EXIST );
if ( $record -> instanceid != 0 ) {
debugging ( 'Invalid system context detected' );
2007-06-26 16:16:46 +00:00
}
2011-10-14 12:48:00 +02:00
if ( defined ( 'SYSCONTEXTID' ) and $record -> id != SYSCONTEXTID ) {
debugging ( 'Invalid SYSCONTEXTID detected' );
2006-09-11 08:56:23 +00:00
}
2011-10-14 12:48:00 +02:00
if ( $record -> depth != 1 or $record -> path != '/' . $record -> id ) {
// fix path if necessary, initial install or path reset
$record -> depth = 1 ;
$record -> path = '/' . $record -> id ;
$DB -> update_record ( 'context' , $record );
2010-03-07 09:28:54 +00:00
}
2011-10-14 12:48:00 +02:00
}
}
/**
* User context class
2012-01-14 19:12:43 +01:00
*
* @ package core_access
* @ category access
* @ copyright Petr Skoda { @ link http :// skodak . org }
* @ license http :// www . gnu . org / copyleft / gpl . html GNU GPL v3 or later
* @ since 2.2
2011-10-14 12:48:00 +02:00
*/
class context_user extends context {
/**
* Please use context_user :: instance ( $userid ) if you need the instance of context .
* Alternatively if you know only the context id use context :: instance_by_id ( $contextid )
*
* @ param stdClass $record
*/
protected function __construct ( stdClass $record ) {
parent :: __construct ( $record );
if ( $record -> contextlevel != CONTEXT_USER ) {
throw new coding_exception ( 'Invalid $record->contextlevel in context_user constructor.' );
2008-05-25 10:16:53 +00:00
}
2008-01-06 23:22:26 +00:00
}
2008-01-06 23:22:08 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns human readable context level name .
*
* @ static
* @ return string the human readable context level name .
*/
2012-01-08 15:39:03 +01:00
public static function get_level_name () {
2011-10-14 12:48:00 +02:00
return get_string ( 'user' );
2008-01-06 23:22:26 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns human readable context identifier .
*
* @ param boolean $withprefix whether to prefix the name of the context with User
* @ param boolean $short does not apply to user context
* @ return string the human readable context name .
*/
public function get_context_name ( $withprefix = true , $short = false ) {
global $DB ;
$name = '' ;
if ( $user = $DB -> get_record ( 'user' , array ( 'id' => $this -> _instanceid , 'deleted' => 0 ))) {
if ( $withprefix ){
$name = get_string ( 'user' ) . ': ' ;
2010-03-07 09:28:54 +00:00
}
2011-10-14 12:48:00 +02:00
$name .= fullname ( $user );
2010-03-07 09:28:54 +00:00
}
2011-10-14 12:48:00 +02:00
return $name ;
2006-09-16 13:54:57 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns the most relevant URL for this context .
*
* @ return moodle_url
*/
public function get_url () {
global $COURSE ;
2006-09-16 13:54:57 +00:00
2011-10-14 12:48:00 +02:00
if ( $COURSE -> id == SITEID ) {
$url = new moodle_url ( '/user/profile.php' , array ( 'id' => $this -> _instanceid ));
2008-01-06 23:22:26 +00:00
} else {
2011-10-14 12:48:00 +02:00
$url = new moodle_url ( '/user/view.php' , array ( 'id' => $this -> _instanceid , 'courseid' => $COURSE -> id ));
2008-01-06 23:22:26 +00:00
}
2011-10-14 12:48:00 +02:00
return $url ;
2008-01-06 23:22:26 +00:00
}
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns array of relevant context capability records .
*
* @ return array
*/
public function get_capabilities () {
global $DB ;
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
$sort = 'ORDER BY contextlevel,component,name' ; // To group them sensibly for display
$extracaps = array ( 'moodle/grade:viewall' );
list ( $extra , $params ) = $DB -> get_in_or_equal ( $extracaps , SQL_PARAMS_NAMED , 'cap' );
$sql = " SELECT *
FROM { capabilities }
WHERE contextlevel = " .CONTEXT_USER. "
OR name $extra " ;
return $records = $DB -> get_records_sql ( $sql . ' ' . $sort , $params );
2008-01-06 23:22:26 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns user context instance .
*
* @ static
* @ param int $instanceid
* @ param int $strictness
* @ return context_user context instance
*/
public static function instance ( $instanceid , $strictness = MUST_EXIST ) {
global $DB ;
2008-01-06 23:23:24 +00:00
2011-10-14 12:48:00 +02:00
if ( $context = context :: cache_get ( CONTEXT_USER , $instanceid )) {
return $context ;
}
2008-01-06 23:23:24 +00:00
2011-10-14 12:48:00 +02:00
if ( ! $record = $DB -> get_record ( 'context' , array ( 'contextlevel' => CONTEXT_USER , 'instanceid' => $instanceid ))) {
if ( $user = $DB -> get_record ( 'user' , array ( 'id' => $instanceid , 'deleted' => 0 ), 'id' , $strictness )) {
$record = context :: insert_context_record ( CONTEXT_USER , $user -> id , '/' . SYSCONTEXTID , 0 );
2011-07-04 18:50:21 +01:00
}
2010-03-07 09:28:54 +00:00
}
2011-10-14 12:48:00 +02:00
if ( $record ) {
$context = new context_user ( $record );
context :: cache_add ( $context );
return $context ;
}
return false ;
2008-05-24 18:35:48 +00:00
}
2008-01-06 23:23:24 +00:00
2011-10-14 12:48:00 +02:00
/**
* Create missing context instances at user context level
* @ static
*/
protected static function create_level_instances () {
global $DB ;
$sql = " INSERT INTO { context} (contextlevel, instanceid)
SELECT " .CONTEXT_USER. " , u . id
FROM { user } u
WHERE u . deleted = 0
AND NOT EXISTS ( SELECT 'x'
FROM { context } cx
WHERE u . id = cx . instanceid AND cx . contextlevel = " .CONTEXT_USER. " ) " ;
$DB -> execute ( $sql );
2010-03-07 09:28:54 +00:00
}
2008-01-06 23:23:24 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns sql necessary for purging of stale context instances .
*
* @ static
* @ return string cleanup SQL
*/
protected static function get_cleanup_sql () {
$sql = "
SELECT c .*
FROM { context } c
LEFT OUTER JOIN { user } u ON ( c . instanceid = u . id AND u . deleted = 0 )
WHERE u . id IS NULL AND c . contextlevel = " .CONTEXT_USER. "
" ;
return $sql ;
}
/**
* Rebuild context paths and depths at user context level .
*
* @ static
2012-02-21 11:48:33 +13:00
* @ param bool $force
2011-10-14 12:48:00 +02:00
*/
protected static function build_paths ( $force ) {
global $DB ;
2012-04-04 13:57:27 +01:00
// First update normal users.
$path = $DB -> sql_concat ( '?' , 'id' );
$pathstart = '/' . SYSCONTEXTID . '/' ;
$params = array ( $pathstart );
if ( $force ) {
$where = " depth <> 2 OR path IS NULL OR path <> ( { $path } ) " ;
$params [] = $pathstart ;
} else {
$where = " depth = 0 OR path IS NULL " ;
}
2011-10-14 12:48:00 +02:00
$sql = " UPDATE { context}
SET depth = 2 ,
2012-04-04 13:57:27 +01:00
path = { $path }
WHERE contextlevel = " . CONTEXT_USER . "
AND ( $where ) " ;
$DB -> execute ( $sql , $params );
2011-10-14 12:48:00 +02:00
}
}
/**
* Course category context class
2012-01-14 19:12:43 +01:00
*
* @ package core_access
* @ category access
* @ copyright Petr Skoda { @ link http :// skodak . org }
* @ license http :// www . gnu . org / copyleft / gpl . html GNU GPL v3 or later
* @ since 2.2
2011-10-14 12:48:00 +02:00
*/
class context_coursecat extends context {
/**
* Please use context_coursecat :: instance ( $coursecatid ) if you need the instance of context .
* Alternatively if you know only the context id use context :: instance_by_id ( $contextid )
*
* @ param stdClass $record
*/
protected function __construct ( stdClass $record ) {
parent :: __construct ( $record );
if ( $record -> contextlevel != CONTEXT_COURSECAT ) {
throw new coding_exception ( 'Invalid $record->contextlevel in context_coursecat constructor.' );
2010-03-07 09:28:54 +00:00
}
2011-10-14 12:48:00 +02:00
}
/**
* Returns human readable context level name .
*
* @ static
* @ return string the human readable context level name .
*/
2012-01-08 15:39:03 +01:00
public static function get_level_name () {
2011-10-14 12:48:00 +02:00
return get_string ( 'category' );
}
/**
* Returns human readable context identifier .
*
* @ param boolean $withprefix whether to prefix the name of the context with Category
* @ param boolean $short does not apply to course categories
* @ return string the human readable context name .
*/
public function get_context_name ( $withprefix = true , $short = false ) {
global $DB ;
2010-10-26 07:51:09 +00:00
2011-10-14 12:48:00 +02:00
$name = '' ;
if ( $category = $DB -> get_record ( 'course_categories' , array ( 'id' => $this -> _instanceid ))) {
if ( $withprefix ){
$name = get_string ( 'category' ) . ': ' ;
}
$name .= format_string ( $category -> name , true , array ( 'context' => $this ));
}
return $name ;
}
/**
* Returns the most relevant URL for this context .
*
* @ return moodle_url
*/
public function get_url () {
return new moodle_url ( '/course/category.php' , array ( 'id' => $this -> _instanceid ));
}
/**
* Returns array of relevant context capability records .
*
* @ return array
*/
public function get_capabilities () {
global $DB ;
$sort = 'ORDER BY contextlevel,component,name' ; // To group them sensibly for display
$params = array ();
$sql = " SELECT *
FROM { capabilities }
WHERE contextlevel IN ( " .CONTEXT_COURSECAT. " , " .CONTEXT_COURSE. " , " .CONTEXT_MODULE. " , " .CONTEXT_BLOCK. " ) " ;
return $DB -> get_records_sql ( $sql . ' ' . $sort , $params );
}
/**
* Returns course category context instance .
*
* @ static
* @ param int $instanceid
* @ param int $strictness
* @ return context_coursecat context instance
*/
public static function instance ( $instanceid , $strictness = MUST_EXIST ) {
global $DB ;
if ( $context = context :: cache_get ( CONTEXT_COURSECAT , $instanceid )) {
return $context ;
}
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
if ( ! $record = $DB -> get_record ( 'context' , array ( 'contextlevel' => CONTEXT_COURSECAT , 'instanceid' => $instanceid ))) {
if ( $category = $DB -> get_record ( 'course_categories' , array ( 'id' => $instanceid ), 'id,parent' , $strictness )) {
if ( $category -> parent ) {
$parentcontext = context_coursecat :: instance ( $category -> parent );
$record = context :: insert_context_record ( CONTEXT_COURSECAT , $category -> id , $parentcontext -> path );
2008-01-06 23:23:58 +00:00
} else {
2011-10-14 12:48:00 +02:00
$record = context :: insert_context_record ( CONTEXT_COURSECAT , $category -> id , '/' . SYSCONTEXTID , 0 );
2008-01-06 23:23:58 +00:00
}
}
}
2011-10-14 12:48:00 +02:00
if ( $record ) {
$context = new context_coursecat ( $record );
context :: cache_add ( $context );
return $context ;
2008-01-06 23:24:25 +00:00
}
2008-01-06 23:23:58 +00:00
2011-10-14 12:48:00 +02:00
return false ;
2010-03-07 09:28:54 +00:00
}
2008-01-06 23:24:25 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns immediate child contexts of category and all subcategories ,
* children of subcategories and courses are not returned .
*
* @ return array
*/
public function get_child_contexts () {
global $DB ;
2008-01-06 23:24:25 +00:00
2011-10-14 12:48:00 +02:00
$sql = " SELECT ctx.*
FROM { context } ctx
WHERE ctx . path LIKE ? AND ( ctx . depth = ? OR ctx . contextlevel = ? ) " ;
$params = array ( $this -> _path . '/%' , $this -> depth + 1 , CONTEXT_COURSECAT );
$records = $DB -> get_records_sql ( $sql , $params );
2006-09-03 07:56:40 +00:00
2011-10-14 12:48:00 +02:00
$result = array ();
foreach ( $records as $record ) {
$result [ $record -> id ] = context :: create_instance_from_record ( $record );
}
2008-01-06 23:25:37 +00:00
2011-10-14 12:48:00 +02:00
return $result ;
2008-01-06 23:25:37 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Create missing context instances at course category context level
* @ static
*/
protected static function create_level_instances () {
global $DB ;
2008-01-06 23:25:37 +00:00
2011-10-14 12:48:00 +02:00
$sql = " INSERT INTO { context} (contextlevel, instanceid)
SELECT " .CONTEXT_COURSECAT. " , cc . id
FROM { course_categories } cc
WHERE NOT EXISTS ( SELECT 'x'
FROM { context } cx
WHERE cc . id = cx . instanceid AND cx . contextlevel = " .CONTEXT_COURSECAT. " ) " ;
$DB -> execute ( $sql );
2008-01-06 23:25:37 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns sql necessary for purging of stale context instances .
*
* @ static
* @ return string cleanup SQL
*/
protected static function get_cleanup_sql () {
$sql = "
SELECT c .*
FROM { context } c
LEFT OUTER JOIN { course_categories } cc ON c . instanceid = cc . id
WHERE cc . id IS NULL AND c . contextlevel = " .CONTEXT_COURSECAT. "
" ;
2008-01-06 23:25:37 +00:00
2011-10-14 12:48:00 +02:00
return $sql ;
}
2008-01-06 23:25:37 +00:00
2011-10-14 12:48:00 +02:00
/**
* Rebuild context paths and depths at course category context level .
*
* @ static
2012-02-21 11:48:33 +13:00
* @ param bool $force
2011-10-14 12:48:00 +02:00
*/
protected static function build_paths ( $force ) {
global $DB ;
if ( $force or $DB -> record_exists_select ( 'context' , " contextlevel = " . CONTEXT_COURSECAT . " AND (depth = 0 OR path IS NULL) " )) {
if ( $force ) {
$ctxemptyclause = $emptyclause = '' ;
} else {
$ctxemptyclause = " AND (ctx.path IS NULL OR ctx.depth = 0) " ;
$emptyclause = " AND ( { context}.path IS NULL OR { context}.depth = 0) " ;
}
$base = '/' . SYSCONTEXTID ;
// Normal top level categories
$sql = " UPDATE { context}
SET depth = 2 ,
path = " . $DB->sql_concat ( " '$base/' " , 'id'). "
WHERE contextlevel = " .CONTEXT_COURSECAT. "
AND EXISTS ( SELECT 'x'
FROM { course_categories } cc
WHERE cc . id = { context } . instanceid AND cc . depth = 1 )
$emptyclause " ;
$DB -> execute ( $sql );
// Deeper categories - one query per depthlevel
$maxdepth = $DB -> get_field_sql ( " SELECT MAX(depth) FROM { course_categories} " );
for ( $n = 2 ; $n <= $maxdepth ; $n ++ ) {
$sql = " INSERT INTO { context_temp} (id, path, depth)
SELECT ctx . id , " . $DB->sql_concat ('pctx.path', " '/' " , 'ctx.id'). " , pctx . depth + 1
FROM { context } ctx
JOIN { course_categories } cc ON ( cc . id = ctx . instanceid AND ctx . contextlevel = " .CONTEXT_COURSECAT. " AND cc . depth = $n )
JOIN { context } pctx ON ( pctx . instanceid = cc . parent AND pctx . contextlevel = " .CONTEXT_COURSECAT. " )
WHERE pctx . path IS NOT NULL AND pctx . depth > 0
$ctxemptyclause " ;
$trans = $DB -> start_delegated_transaction ();
$DB -> delete_records ( 'context_temp' );
$DB -> execute ( $sql );
context :: merge_context_temp_table ();
$DB -> delete_records ( 'context_temp' );
$trans -> allow_commit ();
}
}
2008-01-06 23:25:37 +00:00
}
}
2011-10-14 12:48:00 +02:00
2006-09-12 07:37:23 +00:00
/**
2011-10-14 12:48:00 +02:00
* Course context class
2012-01-14 19:12:43 +01:00
*
* @ package core_access
* @ category access
* @ copyright Petr Skoda { @ link http :// skodak . org }
* @ license http :// www . gnu . org / copyleft / gpl . html GNU GPL v3 or later
* @ since 2.2
2006-09-12 07:37:23 +00:00
*/
2011-10-14 12:48:00 +02:00
class context_course extends context {
/**
* Please use context_course :: instance ( $courseid ) if you need the instance of context .
* Alternatively if you know only the context id use context :: instance_by_id ( $contextid )
*
* @ param stdClass $record
*/
protected function __construct ( stdClass $record ) {
parent :: __construct ( $record );
if ( $record -> contextlevel != CONTEXT_COURSE ) {
throw new coding_exception ( 'Invalid $record->contextlevel in context_course constructor.' );
}
}
2006-09-20 21:00:45 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns human readable context level name .
*
* @ static
* @ return string the human readable context level name .
*/
2012-01-08 15:39:03 +01:00
public static function get_level_name () {
2011-10-14 12:48:00 +02:00
return get_string ( 'course' );
2006-10-25 07:20:28 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns human readable context identifier .
*
* @ param boolean $withprefix whether to prefix the name of the context with Course
* @ param boolean $short whether to use the short name of the thing .
* @ return string the human readable context name .
*/
public function get_context_name ( $withprefix = true , $short = false ) {
global $DB ;
$name = '' ;
if ( $this -> _instanceid == SITEID ) {
$name = get_string ( 'frontpage' , 'admin' );
} else {
if ( $course = $DB -> get_record ( 'course' , array ( 'id' => $this -> _instanceid ))) {
if ( $withprefix ){
$name = get_string ( 'course' ) . ': ' ;
}
if ( $short ){
$name .= format_string ( $course -> shortname , true , array ( 'context' => $this ));
} else {
2012-08-02 17:21:32 +01:00
$name .= format_string ( get_course_display_name_for_list ( $course ));
2011-10-14 12:48:00 +02:00
}
}
2006-09-12 07:37:23 +00:00
}
2011-10-14 12:48:00 +02:00
return $name ;
2006-09-20 21:00:45 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns the most relevant URL for this context .
*
* @ return moodle_url
*/
public function get_url () {
if ( $this -> _instanceid != SITEID ) {
return new moodle_url ( '/course/view.php' , array ( 'id' => $this -> _instanceid ));
}
return new moodle_url ( '/' );
}
/**
* Returns array of relevant context capability records .
*
* @ return array
*/
public function get_capabilities () {
global $DB ;
$sort = 'ORDER BY contextlevel,component,name' ; // To group them sensibly for display
2008-05-24 18:35:48 +00:00
$params = array ();
2011-10-14 12:48:00 +02:00
$sql = " SELECT *
FROM { capabilities }
WHERE contextlevel IN ( " .CONTEXT_COURSE. " , " .CONTEXT_MODULE. " , " .CONTEXT_BLOCK. " ) " ;
return $DB -> get_records_sql ( $sql . ' ' . $sort , $params );
2006-10-01 08:34:36 +00:00
}
2007-09-13 13:44:35 +00:00
2011-10-14 12:48:00 +02:00
/**
* Is this context part of any course ? If yes return course context .
*
* @ param bool $strict true means throw exception if not found , false means return false if not found
* @ return course_context context of the enclosing course , null if not found or exception
*/
public function get_course_context ( $strict = true ) {
return $this ;
}
/**
* Returns course context instance .
*
* @ static
* @ param int $instanceid
* @ param int $strictness
* @ return context_course context instance
*/
public static function instance ( $instanceid , $strictness = MUST_EXIST ) {
global $DB ;
if ( $context = context :: cache_get ( CONTEXT_COURSE , $instanceid )) {
return $context ;
}
if ( ! $record = $DB -> get_record ( 'context' , array ( 'contextlevel' => CONTEXT_COURSE , 'instanceid' => $instanceid ))) {
if ( $course = $DB -> get_record ( 'course' , array ( 'id' => $instanceid ), 'id,category' , $strictness )) {
if ( $course -> category ) {
$parentcontext = context_coursecat :: instance ( $course -> category );
$record = context :: insert_context_record ( CONTEXT_COURSE , $course -> id , $parentcontext -> path );
} else {
$record = context :: insert_context_record ( CONTEXT_COURSE , $course -> id , '/' . SYSCONTEXTID , 0 );
}
}
}
if ( $record ) {
$context = new context_course ( $record );
context :: cache_add ( $context );
return $context ;
}
return false ;
}
/**
* Create missing context instances at course context level
* @ static
*/
protected static function create_level_instances () {
global $DB ;
$sql = " INSERT INTO { context} (contextlevel, instanceid)
SELECT " .CONTEXT_COURSE. " , c . id
FROM { course } c
WHERE NOT EXISTS ( SELECT 'x'
FROM { context } cx
WHERE c . id = cx . instanceid AND cx . contextlevel = " .CONTEXT_COURSE. " ) " ;
$DB -> execute ( $sql );
2007-07-09 06:22:56 +00:00
}
2006-10-01 08:34:36 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns sql necessary for purging of stale context instances .
*
* @ static
* @ return string cleanup SQL
*/
protected static function get_cleanup_sql () {
$sql = "
SELECT c .*
FROM { context } c
LEFT OUTER JOIN { course } co ON c . instanceid = co . id
WHERE co . id IS NULL AND c . contextlevel = " .CONTEXT_COURSE. "
" ;
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
return $sql ;
2008-10-30 10:49:15 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Rebuild context paths and depths at course context level .
*
* @ static
2012-02-21 11:48:33 +13:00
* @ param bool $force
2011-10-14 12:48:00 +02:00
*/
protected static function build_paths ( $force ) {
global $DB ;
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
if ( $force or $DB -> record_exists_select ( 'context' , " contextlevel = " . CONTEXT_COURSE . " AND (depth = 0 OR path IS NULL) " )) {
if ( $force ) {
$ctxemptyclause = $emptyclause = '' ;
} else {
$ctxemptyclause = " AND (ctx.path IS NULL OR ctx.depth = 0) " ;
$emptyclause = " AND ( { context}.path IS NULL OR { context}.depth = 0) " ;
}
$base = '/' . SYSCONTEXTID ;
// Standard frontpage
$sql = " UPDATE { context}
SET depth = 2 ,
path = " . $DB->sql_concat ( " '$base/' " , 'id'). "
WHERE contextlevel = " .CONTEXT_COURSE. "
AND EXISTS ( SELECT 'x'
FROM { course } c
WHERE c . id = { context } . instanceid AND c . category = 0 )
$emptyclause " ;
$DB -> execute ( $sql );
// standard courses
$sql = " INSERT INTO { context_temp} (id, path, depth)
SELECT ctx . id , " . $DB->sql_concat ('pctx.path', " '/' " , 'ctx.id'). " , pctx . depth + 1
FROM { context } ctx
JOIN { course } c ON ( c . id = ctx . instanceid AND ctx . contextlevel = " .CONTEXT_COURSE. " AND c . category <> 0 )
JOIN { context } pctx ON ( pctx . instanceid = c . category AND pctx . contextlevel = " .CONTEXT_COURSECAT. " )
WHERE pctx . path IS NOT NULL AND pctx . depth > 0
$ctxemptyclause " ;
$trans = $DB -> start_delegated_transaction ();
$DB -> delete_records ( 'context_temp' );
$DB -> execute ( $sql );
context :: merge_context_temp_table ();
$DB -> delete_records ( 'context_temp' );
$trans -> allow_commit ();
}
}
2006-09-12 07:37:23 +00:00
}
2011-10-14 12:48:00 +02:00
2006-09-28 04:42:49 +00:00
/**
2011-10-14 12:48:00 +02:00
* Course module context class
2012-01-14 19:12:43 +01:00
*
* @ package core_access
* @ category access
* @ copyright Petr Skoda { @ link http :// skodak . org }
* @ license http :// www . gnu . org / copyleft / gpl . html GNU GPL v3 or later
* @ since 2.2
2006-09-28 04:42:49 +00:00
*/
2011-10-14 12:48:00 +02:00
class context_module extends context {
/**
* Please use context_module :: instance ( $cmid ) if you need the instance of context .
* Alternatively if you know only the context id use context :: instance_by_id ( $contextid )
*
* @ param stdClass $record
*/
protected function __construct ( stdClass $record ) {
parent :: __construct ( $record );
if ( $record -> contextlevel != CONTEXT_MODULE ) {
throw new coding_exception ( 'Invalid $record->contextlevel in context_module constructor.' );
2006-09-28 04:42:49 +00:00
}
}
2011-10-14 12:48:00 +02:00
/**
* Returns human readable context level name .
*
* @ static
* @ return string the human readable context level name .
*/
2012-01-08 15:39:03 +01:00
public static function get_level_name () {
2011-10-14 12:48:00 +02:00
return get_string ( 'activitymodule' );
2008-12-19 08:45:44 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns human readable context identifier .
*
* @ param boolean $withprefix whether to prefix the name of the context with the
* module name , e . g . Forum , Glossary , etc .
* @ param boolean $short does not apply to module context
* @ return string the human readable context name .
*/
public function get_context_name ( $withprefix = true , $short = false ) {
global $DB ;
$name = '' ;
if ( $cm = $DB -> get_record_sql ( " SELECT cm.*, md.name AS modname
FROM { course_modules } cm
JOIN { modules } md ON md . id = cm . module
WHERE cm . id = ? " , array( $this->_instanceid ))) {
if ( $mod = $DB -> get_record ( $cm -> modname , array ( 'id' => $cm -> instance ))) {
if ( $withprefix ){
$name = get_string ( 'modulename' , $cm -> modname ) . ': ' ;
}
2012-08-29 22:25:28 +01:00
$name .= format_string ( $mod -> name , true , array ( 'context' => $this ));
2011-10-14 12:48:00 +02:00
}
}
return $name ;
}
2008-12-19 08:45:44 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns the most relevant URL for this context .
*
* @ return moodle_url
*/
public function get_url () {
global $DB ;
2006-09-28 04:42:49 +00:00
2011-10-14 12:48:00 +02:00
if ( $modname = $DB -> get_field_sql ( " SELECT md.name AS modname
FROM { course_modules } cm
JOIN { modules } md ON md . id = cm . module
WHERE cm . id = ? " , array( $this->_instanceid ))) {
return new moodle_url ( '/mod/' . $modname . '/view.php' , array ( 'id' => $this -> _instanceid ));
}
2006-09-28 04:42:49 +00:00
2011-10-14 12:48:00 +02:00
return new moodle_url ( '/' );
}
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns array of relevant context capability records .
*
* @ return array
*/
public function get_capabilities () {
global $DB , $CFG ;
$sort = 'ORDER BY contextlevel,component,name' ; // To group them sensibly for display
$cm = $DB -> get_record ( 'course_modules' , array ( 'id' => $this -> _instanceid ));
$module = $DB -> get_record ( 'modules' , array ( 'id' => $cm -> module ));
$subcaps = array ();
$subpluginsfile = " $CFG->dirroot /mod/ $module->name /db/subplugins.php " ;
if ( file_exists ( $subpluginsfile )) {
$subplugins = array (); // should be redefined in the file
include ( $subpluginsfile );
if ( ! empty ( $subplugins )) {
foreach ( array_keys ( $subplugins ) as $subplugintype ) {
foreach ( array_keys ( get_plugin_list ( $subplugintype )) as $subpluginname ) {
$subcaps = array_merge ( $subcaps , array_keys ( load_capability_def ( $subplugintype . '_' . $subpluginname )));
}
}
}
2007-05-14 12:11:47 +00:00
}
2011-10-14 12:48:00 +02:00
$modfile = " $CFG->dirroot /mod/ $module->name /lib.php " ;
2012-01-25 15:21:37 +13:00
$extracaps = array ();
2011-10-14 12:48:00 +02:00
if ( file_exists ( $modfile )) {
include_once ( $modfile );
$modfunction = $module -> name . '_get_extra_capabilities' ;
if ( function_exists ( $modfunction )) {
$extracaps = $modfunction ();
2007-05-14 12:11:47 +00:00
}
}
2007-09-13 13:44:35 +00:00
2011-10-14 12:48:00 +02:00
$extracaps = array_merge ( $subcaps , $extracaps );
2012-01-25 15:21:37 +13:00
$extra = '' ;
2012-02-01 23:09:49 +13:00
list ( $extra , $params ) = $DB -> get_in_or_equal (
$extracaps , SQL_PARAMS_NAMED , 'cap0' , true , '' );
if ( ! empty ( $extra )) {
2012-01-25 15:21:37 +13:00
$extra = " OR name $extra " ;
}
2011-10-14 12:48:00 +02:00
$sql = " SELECT *
FROM { capabilities }
WHERE ( contextlevel = " .CONTEXT_MODULE. "
2011-12-18 11:54:02 +01:00
AND ( component = : component OR component = 'moodle' ))
2011-10-14 12:48:00 +02:00
$extra " ;
$params [ 'component' ] = " mod_ $module->name " ;
return $DB -> get_records_sql ( $sql . ' ' . $sort , $params );
2006-09-19 01:44:33 +00:00
}
2006-09-21 03:21:33 +00:00
2011-10-14 12:48:00 +02:00
/**
* Is this context part of any course ? If yes return course context .
*
* @ param bool $strict true means throw exception if not found , false means return false if not found
* @ return course_context context of the enclosing course , null if not found or exception
*/
public function get_course_context ( $strict = true ) {
return $this -> get_parent_context ();
}
2006-09-21 22:34:45 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns module context instance .
*
* @ static
* @ param int $instanceid
* @ param int $strictness
* @ return context_module context instance
*/
public static function instance ( $instanceid , $strictness = MUST_EXIST ) {
global $DB ;
2006-09-21 22:34:45 +00:00
2011-10-14 12:48:00 +02:00
if ( $context = context :: cache_get ( CONTEXT_MODULE , $instanceid )) {
return $context ;
}
2006-09-21 06:57:14 +00:00
2011-10-14 12:48:00 +02:00
if ( ! $record = $DB -> get_record ( 'context' , array ( 'contextlevel' => CONTEXT_MODULE , 'instanceid' => $instanceid ))) {
if ( $cm = $DB -> get_record ( 'course_modules' , array ( 'id' => $instanceid ), 'id,course' , $strictness )) {
$parentcontext = context_course :: instance ( $cm -> course );
$record = context :: insert_context_record ( CONTEXT_MODULE , $cm -> id , $parentcontext -> path );
}
}
2006-09-21 09:16:41 +00:00
2011-10-14 12:48:00 +02:00
if ( $record ) {
$context = new context_module ( $record );
context :: cache_add ( $context );
return $context ;
}
2006-09-21 09:16:41 +00:00
2011-10-14 12:48:00 +02:00
return false ;
2007-09-19 07:11:18 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Create missing context instances at module context level
* @ static
*/
protected static function create_level_instances () {
global $DB ;
$sql = " INSERT INTO { context} (contextlevel, instanceid)
SELECT " .CONTEXT_MODULE. " , cm . id
FROM { course_modules } cm
WHERE NOT EXISTS ( SELECT 'x'
FROM { context } cx
WHERE cm . id = cx . instanceid AND cx . contextlevel = " .CONTEXT_MODULE. " ) " ;
$DB -> execute ( $sql );
2007-09-19 07:06:03 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns sql necessary for purging of stale context instances .
*
* @ static
* @ return string cleanup SQL
*/
protected static function get_cleanup_sql () {
$sql = "
SELECT c .*
FROM { context } c
LEFT OUTER JOIN { course_modules } cm ON c . instanceid = cm . id
WHERE cm . id IS NULL AND c . contextlevel = " .CONTEXT_MODULE. "
" ;
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
return $sql ;
}
2006-09-21 09:16:41 +00:00
2011-10-14 12:48:00 +02:00
/**
* Rebuild context paths and depths at module context level .
*
* @ static
2012-02-21 11:48:33 +13:00
* @ param bool $force
2011-10-14 12:48:00 +02:00
*/
protected static function build_paths ( $force ) {
global $DB ;
2006-09-21 09:16:41 +00:00
2011-10-14 12:48:00 +02:00
if ( $force or $DB -> record_exists_select ( 'context' , " contextlevel = " . CONTEXT_MODULE . " AND (depth = 0 OR path IS NULL) " )) {
if ( $force ) {
$ctxemptyclause = '' ;
} else {
$ctxemptyclause = " AND (ctx.path IS NULL OR ctx.depth = 0) " ;
}
$sql = " INSERT INTO { context_temp} (id, path, depth)
SELECT ctx . id , " . $DB->sql_concat ('pctx.path', " '/' " , 'ctx.id'). " , pctx . depth + 1
FROM { context } ctx
JOIN { course_modules } cm ON ( cm . id = ctx . instanceid AND ctx . contextlevel = " .CONTEXT_MODULE. " )
JOIN { context } pctx ON ( pctx . instanceid = cm . course AND pctx . contextlevel = " .CONTEXT_COURSE. " )
WHERE pctx . path IS NOT NULL AND pctx . depth > 0
$ctxemptyclause " ;
$trans = $DB -> start_delegated_transaction ();
$DB -> delete_records ( 'context_temp' );
$DB -> execute ( $sql );
context :: merge_context_temp_table ();
$DB -> delete_records ( 'context_temp' );
$trans -> allow_commit ();
}
}
2010-08-27 01:44:25 +00:00
}
2006-09-21 09:16:41 +00:00
2011-10-14 12:48:00 +02:00
2008-05-24 18:35:48 +00:00
/**
2011-10-14 12:48:00 +02:00
* Block context class
2012-01-14 19:12:43 +01:00
*
* @ package core_access
* @ category access
* @ copyright Petr Skoda { @ link http :// skodak . org }
* @ license http :// www . gnu . org / copyleft / gpl . html GNU GPL v3 or later
* @ since 2.2
2008-05-24 18:35:48 +00:00
*/
2011-10-14 12:48:00 +02:00
class context_block extends context {
/**
* Please use context_block :: instance ( $blockinstanceid ) if you need the instance of context .
* Alternatively if you know only the context id use context :: instance_by_id ( $contextid )
*
* @ param stdClass $record
*/
protected function __construct ( stdClass $record ) {
parent :: __construct ( $record );
if ( $record -> contextlevel != CONTEXT_BLOCK ) {
throw new coding_exception ( 'Invalid $record->contextlevel in context_block constructor' );
}
}
2006-09-21 22:34:45 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns human readable context level name .
*
* @ static
* @ return string the human readable context level name .
*/
2012-01-08 15:39:03 +01:00
public static function get_level_name () {
2011-10-14 12:48:00 +02:00
return get_string ( 'block' );
}
2006-09-21 06:57:14 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns human readable context identifier .
*
* @ param boolean $withprefix whether to prefix the name of the context with Block
* @ param boolean $short does not apply to block context
* @ return string the human readable context name .
*/
public function get_context_name ( $withprefix = true , $short = false ) {
global $DB , $CFG ;
$name = '' ;
if ( $blockinstance = $DB -> get_record ( 'block_instances' , array ( 'id' => $this -> _instanceid ))) {
global $CFG ;
require_once ( " $CFG->dirroot /blocks/moodleblock.class.php " );
require_once ( " $CFG->dirroot /blocks/ $blockinstance->blockname /block_ $blockinstance->blockname .php " );
$blockname = " block_ $blockinstance->blockname " ;
if ( $blockobject = new $blockname ()) {
if ( $withprefix ){
$name = get_string ( 'block' ) . ': ' ;
}
$name .= $blockobject -> title ;
}
}
2006-09-21 22:34:45 +00:00
2011-10-14 12:48:00 +02:00
return $name ;
}
2006-09-21 09:09:16 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns the most relevant URL for this context .
*
* @ return moodle_url
*/
public function get_url () {
$parentcontexts = $this -> get_parent_context ();
return $parentcontexts -> get_url ();
}
2006-09-23 12:51:00 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns array of relevant context capability records .
*
* @ return array
*/
public function get_capabilities () {
global $DB ;
2006-09-22 01:46:45 +00:00
2011-10-14 12:48:00 +02:00
$sort = 'ORDER BY contextlevel,component,name' ; // To group them sensibly for display
2006-09-22 01:46:45 +00:00
2011-10-14 12:48:00 +02:00
$params = array ();
$bi = $DB -> get_record ( 'block_instances' , array ( 'id' => $this -> _instanceid ));
2006-09-22 01:46:45 +00:00
2011-10-14 12:48:00 +02:00
$extra = '' ;
$extracaps = block_method_result ( $bi -> blockname , 'get_extra_capabilities' );
if ( $extracaps ) {
list ( $extra , $params ) = $DB -> get_in_or_equal ( $extracaps , SQL_PARAMS_NAMED , 'cap' );
$extra = " OR name $extra " ;
}
2006-09-21 22:34:45 +00:00
2011-10-14 12:48:00 +02:00
$sql = " SELECT *
FROM { capabilities }
WHERE ( contextlevel = " .CONTEXT_BLOCK. "
AND component = : component )
$extra " ;
$params [ 'component' ] = 'block_' . $bi -> blockname ;
2006-09-21 09:16:41 +00:00
2011-10-14 12:48:00 +02:00
return $DB -> get_records_sql ( $sql . ' ' . $sort , $params );
}
2006-10-04 07:07:57 +00:00
2011-10-14 12:48:00 +02:00
/**
* Is this context part of any course ? If yes return course context .
*
* @ param bool $strict true means throw exception if not found , false means return false if not found
* @ return course_context context of the enclosing course , null if not found or exception
*/
public function get_course_context ( $strict = true ) {
$parentcontext = $this -> get_parent_context ();
return $parentcontext -> get_course_context ( $strict );
}
/**
* Returns block context instance .
*
* @ static
* @ param int $instanceid
* @ param int $strictness
* @ return context_block context instance
*/
public static function instance ( $instanceid , $strictness = MUST_EXIST ) {
global $DB ;
if ( $context = context :: cache_get ( CONTEXT_BLOCK , $instanceid )) {
return $context ;
2011-01-30 21:49:22 +01:00
}
2011-10-14 12:48:00 +02:00
if ( ! $record = $DB -> get_record ( 'context' , array ( 'contextlevel' => CONTEXT_BLOCK , 'instanceid' => $instanceid ))) {
if ( $bi = $DB -> get_record ( 'block_instances' , array ( 'id' => $instanceid ), 'id,parentcontextid' , $strictness )) {
$parentcontext = context :: instance_by_id ( $bi -> parentcontextid );
$record = context :: insert_context_record ( CONTEXT_BLOCK , $bi -> id , $parentcontext -> path );
}
}
2011-01-30 21:49:22 +01:00
2011-10-14 12:48:00 +02:00
if ( $record ) {
$context = new context_block ( $record );
context :: cache_add ( $context );
return $context ;
}
2011-01-30 21:49:22 +01:00
2011-10-14 12:48:00 +02:00
return false ;
2006-10-04 07:07:57 +00:00
}
2007-01-12 07:41:33 +00:00
2011-10-14 12:48:00 +02:00
/**
* Block do not have child contexts ...
* @ return array
*/
public function get_child_contexts () {
return array ();
}
2008-05-24 18:35:48 +00:00
2011-10-14 12:48:00 +02:00
/**
* Create missing context instances at block context level
* @ static
*/
protected static function create_level_instances () {
global $DB ;
$sql = " INSERT INTO { context} (contextlevel, instanceid)
SELECT " .CONTEXT_BLOCK. " , bi . id
FROM { block_instances } bi
WHERE NOT EXISTS ( SELECT 'x'
FROM { context } cx
WHERE bi . id = cx . instanceid AND cx . contextlevel = " .CONTEXT_BLOCK. " ) " ;
$DB -> execute ( $sql );
2008-02-21 09:20:20 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Returns sql necessary for purging of stale context instances .
*
* @ static
* @ return string cleanup SQL
*/
protected static function get_cleanup_sql () {
$sql = "
SELECT c .*
FROM { context } c
LEFT OUTER JOIN { block_instances } bi ON c . instanceid = bi . id
WHERE bi . id IS NULL AND c . contextlevel = " .CONTEXT_BLOCK. "
" ;
2008-11-13 08:11:10 +00:00
2011-10-14 12:48:00 +02:00
return $sql ;
2008-11-13 08:11:10 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Rebuild context paths and depths at block context level .
*
* @ static
2012-02-21 11:48:33 +13:00
* @ param bool $force
2011-10-14 12:48:00 +02:00
*/
protected static function build_paths ( $force ) {
global $DB ;
2008-11-13 08:11:10 +00:00
2011-10-14 12:48:00 +02:00
if ( $force or $DB -> record_exists_select ( 'context' , " contextlevel = " . CONTEXT_BLOCK . " AND (depth = 0 OR path IS NULL) " )) {
if ( $force ) {
$ctxemptyclause = '' ;
} else {
$ctxemptyclause = " AND (ctx.path IS NULL OR ctx.depth = 0) " ;
2008-02-21 09:20:20 +00:00
}
2008-11-13 08:11:10 +00:00
2011-10-14 12:48:00 +02:00
// pctx.path IS NOT NULL prevents fatal problems with broken block instances that point to invalid context parent
$sql = " INSERT INTO { context_temp} (id, path, depth)
SELECT ctx . id , " . $DB->sql_concat ('pctx.path', " '/' " , 'ctx.id'). " , pctx . depth + 1
FROM { context } ctx
JOIN { block_instances } bi ON ( bi . id = ctx . instanceid AND ctx . contextlevel = " .CONTEXT_BLOCK. " )
JOIN { context } pctx ON ( pctx . id = bi . parentcontextid )
WHERE ( pctx . path IS NOT NULL AND pctx . depth > 0 )
$ctxemptyclause " ;
$trans = $DB -> start_delegated_transaction ();
$DB -> delete_records ( 'context_temp' );
$DB -> execute ( $sql );
context :: merge_context_temp_table ();
$DB -> delete_records ( 'context_temp' );
$trans -> allow_commit ();
2008-11-13 08:11:10 +00:00
}
2007-03-22 08:11:30 +00:00
}
}
2007-08-02 08:28:29 +00:00
2011-10-14 12:48:00 +02:00
2011-10-25 14:20:22 +02:00
// ============== DEPRECATED FUNCTIONS ==========================================
// Old context related functions were deprecated in 2.0, it is recommended
// to use context classes in new code. Old function can be used when
// creating patches that are supposed to be backported to older stable branches.
// These deprecated functions will not be removed in near future,
// before removing devs will be warned with a debugging message first,
// then we will add error message and only after that we can remove the functions
// completely.
2011-10-14 12:48:00 +02:00
2007-08-16 08:48:53 +00:00
/**
2011-10-25 14:20:22 +02:00
* Not available any more , use load_temp_course_role () instead .
*
* @ deprecated since 2.2
2011-10-14 12:48:00 +02:00
* @ param stdClass $context
* @ param int $roleid
* @ param array $accessdata
* @ return array
2007-08-16 08:48:53 +00:00
*/
2011-10-14 12:48:00 +02:00
function load_temp_role ( $context , $roleid , array $accessdata ) {
debugging ( 'load_temp_role() is deprecated, please use load_temp_course_role() instead, temp role not loaded.' );
return $accessdata ;
}
2007-08-16 08:48:53 +00:00
2011-10-14 12:48:00 +02:00
/**
2011-10-25 14:20:22 +02:00
* Not available any more , use remove_temp_course_roles () instead .
*
* @ deprecated since 2.2
2012-01-14 19:12:43 +01:00
* @ param stdClass $context
2011-10-14 12:48:00 +02:00
* @ param array $accessdata
* @ return array access data
*/
function remove_temp_roles ( $context , array $accessdata ) {
debugging ( 'remove_temp_role() is deprecated, please use remove_temp_course_roles() instead.' );
return $accessdata ;
}
2007-08-16 08:48:53 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns system context or null if can not be created yet .
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use context_system :: instance ()
2011-10-14 12:48:00 +02:00
* @ param bool $cache use caching
* @ return context system context ( null if context table not created yet )
*/
function get_system_context ( $cache = true ) {
return context_system :: instance ( 0 , IGNORE_MISSING , $cache );
}
2007-08-16 08:48:53 +00:00
2011-10-14 12:48:00 +02:00
/**
* Get the context instance as an object . This function will create the
* context instance if it does not exist yet .
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use context_course :: instance () or other relevant class instead
2011-10-14 12:48:00 +02:00
* @ param integer $contextlevel The context level , for example CONTEXT_COURSE , or CONTEXT_MODULE .
* @ param integer $instance The instance id . For $level = CONTEXT_COURSE , this would be $course -> id ,
* for $level = CONTEXT_MODULE , this would be $cm -> id . And so on . Defaults to 0
* @ param int $strictness IGNORE_MISSING means compatible mode , false returned if record not found , debug message if more found ;
* MUST_EXIST means throw exception if no record or multiple records found
* @ return context The context object .
*/
function get_context_instance ( $contextlevel , $instance = 0 , $strictness = IGNORE_MISSING ) {
$instances = ( array ) $instance ;
$contexts = array ();
2008-11-27 20:30:14 +00:00
2011-10-14 12:48:00 +02:00
$classname = context_helper :: get_class_for_level ( $contextlevel );
// we do not load multiple contexts any more, PAGE should be responsible for any preloading
foreach ( $instances as $inst ) {
$contexts [ $inst ] = $classname :: instance ( $inst , $strictness );
2007-08-16 08:48:53 +00:00
}
2011-10-14 12:48:00 +02:00
if ( is_array ( $instance )) {
return $contexts ;
} else {
return $contexts [ $instance ];
}
2007-09-19 07:01:27 +00:00
}
2008-06-16 14:25:53 +00:00
/**
2011-10-14 12:48:00 +02:00
* Get a context instance as an object , from a given context id .
2009-05-22 02:03:55 +00:00
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use context :: instance_by_id ( $id ) instead
2011-10-14 12:48:00 +02:00
* @ param int $id context id
* @ param int $strictness IGNORE_MISSING means compatible mode , false returned if record not found , debug message if more found ;
* MUST_EXIST means throw exception if no record or multiple records found
* @ return context | bool the context object or false if not found .
2008-06-16 14:25:53 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_context_instance_by_id ( $id , $strictness = IGNORE_MISSING ) {
return context :: instance_by_id ( $id , $strictness );
2008-06-16 14:25:53 +00:00
}
2007-09-19 07:01:27 +00:00
/**
2011-10-14 12:48:00 +02:00
* Recursive function which , given a context , find all parent context ids ,
* and return the array in reverse order , i . e . parent first , then grand
* parent , etc .
2009-05-19 07:50:54 +00:00
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use $context -> get_parent_context_ids () instead
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ param bool $includeself optional , defaults to false
* @ return array
2007-09-19 07:01:27 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_parent_contexts ( context $context , $includeself = false ) {
return $context -> get_parent_context_ids ( $includeself );
}
2007-09-19 15:51:04 +00:00
2011-10-14 12:48:00 +02:00
/**
* Return the id of the parent of this context , or false if there is no parent ( only happens if this
* is the site context . )
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use $context -> get_parent_context () instead
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ return integer the id of the parent context .
*/
function get_parent_contextid ( context $context ) {
if ( $parent = $context -> get_parent_context ()) {
return $parent -> id ;
2008-06-22 20:42:19 +00:00
} else {
2011-10-14 12:48:00 +02:00
return false ;
2007-09-19 07:01:27 +00:00
}
2007-08-16 08:48:53 +00:00
}
2007-09-19 07:13:33 +00:00
/**
2011-10-14 12:48:00 +02:00
* Recursive function which , given a context , find all its children context ids .
2009-05-19 07:50:54 +00:00
*
2011-10-14 12:48:00 +02:00
* For course category contexts it will return immediate children only categories and courses .
* It will NOT recurse into courses or child categories .
* If you want to do that , call it on the returned courses / categories .
2007-09-19 07:13:33 +00:00
*
2011-10-14 12:48:00 +02:00
* When called for a course context , it will return the modules and blocks
* displayed in the course page .
*
* If called on a user / course / module context it _will_ populate the cache with the appropriate
* contexts ; - )
2007-09-19 07:13:33 +00:00
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use $context -> get_child_contexts () instead
2012-02-21 11:48:33 +13:00
* @ param context $context
2011-10-14 12:48:00 +02:00
* @ return array Array of child records
*/
function get_child_contexts ( context $context ) {
return $context -> get_child_contexts ();
}
/**
* Precreates all contexts including all parents
2007-09-19 07:13:33 +00:00
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2
2011-10-14 12:48:00 +02:00
* @ param int $contextlevel empty means all
* @ param bool $buildpaths update paths and depths
* @ return void
2007-09-19 07:13:33 +00:00
*/
2011-10-14 12:48:00 +02:00
function create_contexts ( $contextlevel = null , $buildpaths = true ) {
context_helper :: create_instances ( $contextlevel , $buildpaths );
}
2007-09-19 07:13:33 +00:00
2011-10-14 12:48:00 +02:00
/**
* Remove stale context records
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use context_helper :: cleanup_instances () instead
2011-10-14 12:48:00 +02:00
* @ return bool
*/
function cleanup_contexts () {
context_helper :: cleanup_instances ();
return true ;
}
2007-09-19 07:13:33 +00:00
2011-10-14 12:48:00 +02:00
/**
* Populate context . path and context . depth where missing .
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use context_helper :: build_all_paths () instead
2011-10-14 12:48:00 +02:00
* @ param bool $force force a complete rebuild of the path and depth fields , defaults to false
* @ return void
*/
function build_context_path ( $force = false ) {
context_helper :: build_all_paths ( $force );
}
2007-09-19 07:13:33 +00:00
2011-10-14 12:48:00 +02:00
/**
* Rebuild all related context depth and path caches
*
2012-01-14 19:12:43 +01:00
* @ deprecated since 2.2
2011-10-14 12:48:00 +02:00
* @ param array $fixcontexts array of contexts , strongtyped
* @ return void
*/
function rebuild_contexts ( array $fixcontexts ) {
foreach ( $fixcontexts as $fixcontext ) {
$fixcontext -> reset_paths ( false );
}
context_helper :: build_all_paths ( false );
}
2007-09-19 07:13:33 +00:00
2011-10-14 12:48:00 +02:00
/**
* Preloads all contexts relating to a course : course , modules . Block contexts
* are no longer loaded here . The contexts for all the blocks on the current
* page are now efficiently loaded by { @ link block_manager :: load_blocks ()} .
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2
2011-10-14 12:48:00 +02:00
* @ param int $courseid Course ID
* @ return void
*/
function preload_course_contexts ( $courseid ) {
context_helper :: preload_course ( $courseid );
2007-09-19 07:13:33 +00:00
}
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
/**
2010-03-31 07:41:31 +00:00
* Preloads context information together with instances .
2011-10-14 12:48:00 +02:00
* Use context_instance_preload () to strip the context info from the record and cache the context instance .
2010-03-31 07:41:31 +00:00
*
2012-01-14 19:12:43 +01:00
* @ deprecated since 2.2
2010-03-31 07:41:31 +00:00
* @ param string $joinon for example 'u.id'
* @ param string $contextlevel context level of instance in $joinon
* @ param string $tablealias context table alias
* @ return array with two values - select and join part
*/
function context_instance_preload_sql ( $joinon , $contextlevel , $tablealias ) {
2011-10-14 12:48:00 +02:00
$select = " , " . context_helper :: get_preload_record_columns_sql ( $tablealias );
2010-03-31 07:41:31 +00:00
$join = " LEFT JOIN { context} $tablealias ON ( $tablealias .instanceid = $joinon AND $tablealias .contextlevel = $contextlevel ) " ;
return array ( $select , $join );
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
}
/**
2010-03-31 07:41:31 +00:00
* Preloads context information from db record and strips the cached info .
2011-10-14 12:48:00 +02:00
* The db request has to contain both the $join and $select from context_instance_preload_sql ()
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
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2
2011-10-14 12:48:00 +02:00
* @ param stdClass $rec
2010-03-31 07:41:31 +00:00
* @ return void ( modifies $rec )
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
*/
2010-03-31 07:41:31 +00:00
function context_instance_preload ( stdClass $rec ) {
2011-10-14 12:48:00 +02:00
context_helper :: preload_from_record ( $rec );
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
}
2007-10-09 12:49:54 +00:00
/**
2011-10-14 12:48:00 +02:00
* Mark a context as dirty ( with timestamp ) so as to force reloading of the context .
2007-09-19 07:10:09 +00:00
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use $context -> mark_dirty () instead
2007-10-09 12:49:54 +00:00
* @ param string $path context path
2007-09-19 07:10:09 +00:00
*/
function mark_context_dirty ( $path ) {
2011-10-14 12:48:00 +02:00
global $CFG , $USER , $ACCESSLIB_PRIVATE ;
2009-01-12 16:52:53 +00:00
2009-06-24 09:17:56 +00:00
if ( during_initial_install ()) {
2009-01-12 16:52:53 +00:00
return ;
}
2007-09-19 07:22:49 +00:00
// only if it is a non-empty string
if ( is_string ( $path ) && $path !== '' ) {
2007-10-02 08:46:27 +00:00
set_cache_flag ( 'accesslib/dirtycontexts' , $path , 1 , time () + $CFG -> sessiontimeout );
2009-03-23 03:54:50 +00:00
if ( isset ( $ACCESSLIB_PRIVATE -> dirtycontexts )) {
$ACCESSLIB_PRIVATE -> dirtycontexts [ $path ] = 1 ;
2011-10-14 12:48:00 +02:00
} else {
if ( CLI_SCRIPT ) {
$ACCESSLIB_PRIVATE -> dirtycontexts = array ( $path => 1 );
} else {
if ( isset ( $USER -> access [ 'time' ])) {
$ACCESSLIB_PRIVATE -> dirtycontexts = get_cache_flags ( 'accesslib/dirtycontexts' , $USER -> access [ 'time' ] - 2 );
} else {
$ACCESSLIB_PRIVATE -> dirtycontexts = array ( $path => 1 );
}
// flags not loaded yet, it will be done later in $context->reload_if_dirty()
}
2007-10-09 12:49:54 +00:00
}
2007-09-19 07:22:49 +00:00
}
2007-09-19 07:10:09 +00:00
}
2007-10-09 12:49:54 +00:00
/**
2011-10-14 12:48:00 +02:00
* Update the path field of the context and all dep . subcontexts that follow
2008-05-24 18:35:48 +00:00
*
2011-10-14 12:48:00 +02:00
* Update the path field of the context and
* all the dependent subcontexts that follow
* the move .
2007-12-18 20:51:07 +00:00
*
2011-10-14 12:48:00 +02:00
* The most important thing here is to be as
* DB efficient as possible . This op can have a
* massive impact in the DB .
2007-12-18 20:51:07 +00:00
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2
2011-10-14 12:48:00 +02:00
* @ param context $context context obj
* @ param context $newparent new parent obj
* @ return void
2007-12-18 20:51:07 +00:00
*/
2011-10-14 12:48:00 +02:00
function context_moved ( context $context , context $newparent ) {
$context -> update_moved ( $newparent );
2008-05-24 18:35:48 +00:00
}
2010-03-07 09:28:54 +00:00
/**
2011-10-14 12:48:00 +02:00
* Remove a context record and any dependent entries ,
* removes context from static context cache too
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use $context -> delete_content () instead
2011-10-14 12:48:00 +02:00
* @ param int $contextlevel
* @ param int $instanceid
* @ param bool $deleterecord false means keep record for now
* @ return bool returns true or throws an exception
*/
function delete_context ( $contextlevel , $instanceid , $deleterecord = true ) {
if ( $deleterecord ) {
context_helper :: delete_instance ( $contextlevel , $instanceid );
} else {
$classname = context_helper :: get_class_for_level ( $contextlevel );
if ( $context = $classname :: instance ( $instanceid , IGNORE_MISSING )) {
$context -> delete_content ();
2010-03-07 09:28:54 +00:00
}
}
2011-10-14 12:48:00 +02:00
return true ;
2010-03-07 09:28:54 +00:00
}
2010-08-31 07:59:32 +00:00
/**
2011-10-25 14:20:22 +02:00
* Returns context level name
2012-01-14 19:12:43 +01:00
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2
2011-10-14 12:48:00 +02:00
* @ param integer $contextlevel $context -> context level . One of the CONTEXT_ ... constants .
* @ return string the name for this type of context .
2010-08-31 07:59:32 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_contextlevel_name ( $contextlevel ) {
return context_helper :: get_level_name ( $contextlevel );
2010-08-31 07:59:32 +00:00
}
/**
2011-10-14 12:48:00 +02:00
* Prints human readable context identifier .
2010-08-31 07:59:32 +00:00
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2
2011-10-14 12:48:00 +02:00
* @ param context $context the context .
* @ param boolean $withprefix whether to prefix the name of the context with the
* type of context , e . g . User , Course , Forum , etc .
* @ param boolean $short whether to user the short name of the thing . Only applies
* to course contexts
* @ return string the human readable context name .
2010-08-31 07:59:32 +00:00
*/
2011-10-14 12:48:00 +02:00
function print_context_name ( context $context , $withprefix = true , $short = false ) {
return $context -> get_context_name ( $withprefix , $short );
}
2010-08-31 07:59:32 +00:00
2011-10-14 12:48:00 +02:00
/**
* Get a URL for a context , if there is a natural one . For example , for
* CONTEXT_COURSE , this is the course page . For CONTEXT_USER it is the
* user profile page .
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2
2011-10-14 12:48:00 +02:00
* @ param context $context the context .
* @ return moodle_url
*/
function get_context_url ( context $context ) {
return $context -> get_url ();
}
2010-08-31 07:59:32 +00:00
2011-10-14 12:48:00 +02:00
/**
* Is this context part of any course ? if yes return course context ,
* if not return null or throw exception .
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use $context -> get_course_context () instead
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ return course_context context of the enclosing course , null if not found or exception
*/
function get_course_context ( context $context ) {
return $context -> get_course_context ( true );
}
2010-08-31 07:59:32 +00:00
2011-10-14 12:48:00 +02:00
/**
* Returns current course id or null if outside of course based on context parameter .
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use $context -> get_course_context instead
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ return int | bool related course id or false
*/
function get_courseid_from_context ( context $context ) {
if ( $coursecontext = $context -> get_course_context ( false )) {
return $coursecontext -> instanceid ;
} else {
return false ;
2010-08-31 07:59:32 +00:00
}
}
2010-03-07 09:28:54 +00:00
/**
2011-10-14 12:48:00 +02:00
* Get an array of courses where cap requested is available
* and user is enrolled , this can be relatively slow .
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , use enrol_get_users_courses () instead
2011-10-14 12:48:00 +02:00
* @ param int $userid A user id . By default ( null ) checks the permissions of the current user .
* @ param string $cap - name of the capability
* @ param array $accessdata_ignored
* @ param bool $doanything_ignored
* @ param string $sort - sorting fields - prefix each fieldname with " c. "
* @ param array $fields - additional fields you are interested in ...
* @ param int $limit_ignored
* @ return array $courses - ordered array of course objects - see notes above
2010-03-07 09:28:54 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_user_courses_bycap ( $userid , $cap , $accessdata_ignored , $doanything_ignored , $sort = 'c.sortorder ASC' , $fields = null , $limit_ignored = 0 ) {
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
$courses = enrol_get_users_courses ( $userid , true , $fields , $sort );
foreach ( $courses as $id => $course ) {
$context = context_course :: instance ( $id );
if ( ! has_capability ( $cap , $context , $userid )) {
unset ( $courses [ $id ]);
}
}
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
return $courses ;
}
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
/**
* Extracts the relevant capabilities given a contextid .
* All case based , example an instance of forum context .
* Will fetch all forum related capabilities , while course contexts
* Will fetch all capabilities
*
* capabilities
* `name` varchar ( 150 ) NOT NULL ,
* `captype` varchar ( 50 ) NOT NULL ,
* `contextlevel` int ( 10 ) NOT NULL ,
* `component` varchar ( 100 ) NOT NULL ,
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ return array
*/
function fetch_context_capabilities ( context $context ) {
return $context -> get_capabilities ();
}
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
/**
* Runs get_records select on context table and returns the result
* Does get_records_select on the context table , and returns the results ordered
* by contextlevel , and then the natural sort order within each level .
* for the purpose of $select , you need to know that the context table has been
* aliased to ctx , so for example , you can call get_sorted_contexts ( 'ctx.depth = 3' );
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2
2011-10-14 12:48:00 +02:00
* @ param string $select the contents of the WHERE clause . Remember to do ctx . fieldname .
* @ param array $params any parameters required by $select .
* @ return array the requested context records .
*/
function get_sorted_contexts ( $select , $params = array ()) {
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
//TODO: we should probably rewrite all the code that is using this thing, the trouble is we MUST NOT modify the context instances...
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
global $DB ;
if ( $select ) {
$select = 'WHERE ' . $select ;
}
return $DB -> get_records_sql ( "
SELECT ctx .*
FROM { context } ctx
LEFT JOIN { user } u ON ctx . contextlevel = " . CONTEXT_USER . " AND u . id = ctx . instanceid
LEFT JOIN { course_categories } cat ON ctx . contextlevel = " . CONTEXT_COURSECAT . " AND cat . id = ctx . instanceid
LEFT JOIN { course } c ON ctx . contextlevel = " . CONTEXT_COURSE . " AND c . id = ctx . instanceid
LEFT JOIN { course_modules } cm ON ctx . contextlevel = " . CONTEXT_MODULE . " AND cm . id = ctx . instanceid
LEFT JOIN { block_instances } bi ON ctx . contextlevel = " . CONTEXT_BLOCK . " AND bi . id = ctx . instanceid
$select
ORDER BY ctx . contextlevel , bi . defaultregion , COALESCE ( cat . sortorder , c . sortorder , cm . section , bi . defaultweight ), u . lastname , u . firstname , cm . id
" , $params );
2010-03-07 09:28:54 +00:00
}
/**
2011-10-14 12:48:00 +02:00
* This is really slow !!! do not use above course context level
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2
2010-03-07 09:28:54 +00:00
* @ param int $roleid
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ return array
2010-03-07 09:28:54 +00:00
*/
2011-10-14 12:48:00 +02:00
function get_role_context_caps ( $roleid , context $context ) {
2010-03-07 09:28:54 +00:00
global $DB ;
2011-10-14 12:48:00 +02:00
//this is really slow!!!! - do not use above course context level!
$result = array ();
$result [ $context -> id ] = array ();
2010-03-07 09:28:54 +00:00
2011-10-14 12:48:00 +02:00
// first emulate the parent context capabilities merging into context
$searchcontexts = array_reverse ( $context -> get_parent_context_ids ( true ));
foreach ( $searchcontexts as $cid ) {
if ( $capabilities = $DB -> get_records ( 'role_capabilities' , array ( 'roleid' => $roleid , 'contextid' => $cid ))) {
foreach ( $capabilities as $cap ) {
if ( ! array_key_exists ( $cap -> capability , $result [ $context -> id ])) {
$result [ $context -> id ][ $cap -> capability ] = 0 ;
}
$result [ $context -> id ][ $cap -> capability ] += $cap -> permission ;
2010-03-07 09:28:54 +00:00
}
}
2011-10-14 12:48:00 +02:00
}
2010-03-07 09:28:54 +00:00
2012-02-10 02:44:27 +08:00
// now go through the contexts below given context
2011-10-14 12:48:00 +02:00
$searchcontexts = array_keys ( $context -> get_child_contexts ());
foreach ( $searchcontexts as $cid ) {
if ( $capabilities = $DB -> get_records ( 'role_capabilities' , array ( 'roleid' => $roleid , 'contextid' => $cid ))) {
foreach ( $capabilities as $cap ) {
if ( ! array_key_exists ( $cap -> contextid , $result )) {
$result [ $cap -> contextid ] = array ();
}
$result [ $cap -> contextid ][ $cap -> capability ] = $cap -> permission ;
}
2010-03-07 09:28:54 +00:00
}
}
2011-10-14 12:48:00 +02:00
return $result ;
2010-03-07 09:28:54 +00:00
}
2011-10-14 12:48:00 +02:00
/**
* Gets a string for sql calls , searching for stuff in this context or above
*
* NOTE : use $DB -> get_in_or_equal ( $context -> get_parent_context_ids () ...
*
2011-10-25 14:20:22 +02:00
* @ deprecated since 2.2 , $context -> use get_parent_context_ids () instead
2011-10-14 12:48:00 +02:00
* @ param context $context
* @ return string
*/
function get_related_contexts_string ( context $context ) {
if ( $parents = $context -> get_parent_context_ids ()) {
return ( ' IN (' . $context -> id . ',' . implode ( ',' , $parents ) . ')' );
} else {
return ( ' =' . $context -> id );
}
}
2013-03-27 10:14:10 +08:00
/**
* Given context and array of users , returns array of users whose enrolment status is suspended ,
* or enrolment has expired or has not started . Also removes those users from the given array
*
* @ param context $context context in which suspended users should be extracted .
* @ param array $users list of users .
* @ param array $ignoreusers array of user ids to ignore , e . g . guest
* @ return array list of suspended users .
*/
function extract_suspended_users ( $context , & $users , $ignoreusers = array ()) {
global $DB ;
// Get active enrolled users.
list ( $sql , $params ) = get_enrolled_sql ( $context , null , null , true );
$activeusers = $DB -> get_records_sql ( $sql , $params );
// Move suspended users to a separate array & remove from the initial one.
$susers = array ();
if ( sizeof ( $activeusers )) {
foreach ( $users as $userid => $user ) {
if ( ! array_key_exists ( $userid , $activeusers ) && ! in_array ( $userid , $ignoreusers )) {
$susers [ $userid ] = $user ;
unset ( $users [ $userid ]);
}
}
}
return $susers ;
}
/**
* Given context and array of users , returns array of user ids whose enrolment status is suspended ,
* or enrolment has expired or not started .
*
* @ param context $context context in which user enrolment is checked .
* @ return array list of suspended user id ' s .
*/
function get_suspended_userids ( $context ){
global $DB ;
// Get all enrolled users.
list ( $sql , $params ) = get_enrolled_sql ( $context );
$users = $DB -> get_records_sql ( $sql , $params );
// Get active enrolled users.
list ( $sql , $params ) = get_enrolled_sql ( $context , null , null , true );
$activeusers = $DB -> get_records_sql ( $sql , $params );
$susers = array ();
if ( sizeof ( $activeusers ) != sizeof ( $users )) {
foreach ( $users as $userid => $user ) {
if ( ! array_key_exists ( $userid , $activeusers )) {
$susers [ $userid ] = $userid ;
}
}
}
return $susers ;
}