course categories: MDL-17502 when deleting a category and its contents, check moodle/course:delete capability.

* Note: this would never lead to problems with default role definions.
* Also ended up mostly rewriting delete_category_form to simplify the messages that are displayed.
* New helper function require_all_capabilities, a bit like require_any_capability.
This commit is contained in:
tjhunt 2008-12-08 07:28:19 +00:00
parent fc11edbfa0
commit 8a1b1c328d
6 changed files with 119 additions and 42 deletions

View file

@ -1,63 +1,116 @@
<?php //$Id$ <?php //$Id$
require_once($CFG->libdir.'/formslib.php'); require_once($CFG->libdir.'/formslib.php');
require_once($CFG->libdir.'/questionlib.php');
class delete_category_form extends moodleform { class delete_category_form extends moodleform {
var $_category; var $_category;
function definition() { function definition() {
global $CFG; global $CFG, $DB;
$mform =& $this->_form; $mform =& $this->_form;
$category = $this->_customdata; $category = $this->_customdata;
ensure_context_subobj_present($category, CONTEXT_COURSECAT); ensure_context_subobj_present($category, CONTEXT_COURSECAT);
$this->_category = $category; $this->_category = $category;
$mform->addElement('header','general', get_string('categorycurrentcontents', '', format_string($category->name))); /// Check permissions, to see if it OK to give the option to delete
/// the contents, rather than move elsewhere.
$displaylist = array(); /// Are there any subcategories of this one, can they be deleted?
$notused = array();
make_categories_list($displaylist, $notused, 'moodle/course:create', $category->id);
// Check permissions, to see if it OK to give the option to delete
// the contents, rather than move elsewhere.
$candeletecontent = true; $candeletecontent = true;
$tocheck = array($category); $tocheck = get_child_categories($category->id);
$containscategories = !empty($tocheck);
$categoryids = array($category->id);
while (!empty($tocheck)) { while (!empty($tocheck)) {
$checkcat = array_pop($tocheck); $checkcat = array_pop($tocheck);
$childcategoryids[] = $checkcat->id;
$tocheck = $tocheck + get_child_categories($checkcat->id); $tocheck = $tocheck + get_child_categories($checkcat->id);
if (!has_capability('moodle/category:manage', $checkcat->context)) { if ($candeletecontent && !has_capability('moodle/category:manage', $checkcat->context)) {
$candeletecontent = false; $candeletecontent = false;
break;
} }
} }
// TODO check that the user is allowed to delete all the courses MDL-17502! /// Are there any courses in here, can they be deleted?
list($test, $params) = $DB->get_in_or_equal($categoryids);
$containedcourses = $DB->get_records_sql(
"SELECT id,1 FROM {course} c WHERE c.category $test", $params);
$containscourses = false;
if ($containedcourses) {
$containscourses = true;
foreach ($containedcourses as $courseid => $notused) {
if ($candeletecontent && !can_delete_course($courseid)) {
$candeletecontent = false;
break;
}
}
}
/// Are there any questions in the question bank here?
$containsquestions = question_context_has_any_questions($category->context);
/// Get the list of categories we might be able to move to.
$testcaps = array();
if ($containscourses) {
$testcaps[] = 'moodle/course:create';
}
if ($containscategories || $containsquestions) {
$testcaps[] = 'moodle/category:manage';
}
$displaylist = array();
$notused = array();
if (!empty($testcaps)) {
make_categories_list($displaylist, $notused, $testcaps, $category->id);
}
/// Now build the options.
$options = array(); $options = array();
if ($displaylist) { if ($displaylist) {
$options[0] = get_string('move'); $options[0] = get_string('movecontentstoanothercategory');
} }
if ($candeletecontent) { if ($candeletecontent) {
$options[1] = get_string('delete'); $options[1] = get_string('deleteallcannotundo');
} }
if (empty($options)) { /// Now build the form.
print_error('nocategorydelete', 'error', 'index.php', format_string($category->name)); $mform->addElement('header','general', get_string('categorycurrentcontents', '', format_string($category->name)));
}
$mform->addElement('select', 'fulldelete', get_string('categorycontents'), $options); if ($containscourses || $containscategories || $containsquestions) {
$mform->disabledIf('newparent', 'fulldelete', 'eq', '1'); if (empty($options)) {
$mform->setDefault('newparent', 0); print_error('youcannotdeletecategory', 'error', 'index.php', format_string($category->name));
if ($displaylist) {
$mform->addElement('select', 'newparent', get_string('movecategorycontentto'), $displaylist);
if (in_array($category->parent, $displaylist)) {
$mform->setDefault('newparent', $category->parent);
} }
/// Describe the contents of this category.
$contents = '<ul>';
if ($containscategories) {
$contents .= '<li>' . get_string('subcategories') . '</li>';
}
if ($containscourses) {
$contents .= '<li>' . get_string('courses') . '</li>';
}
if ($containsquestions) {
$contents .= '<li>' . get_string('questionsinthequestionbank') . '</li>';
}
$contents .= '</ul>';
$mform->addElement('static', 'emptymessage', get_string('thiscategorycontains'), $contents);
/// Give the options for what to do.
$mform->addElement('select', 'fulldelete', get_string('whattodo'), $options);
if (count($options) == 1) {
$mform->hardFreeze('fulldelete');
$mform->setConstant('fulldelete', reset(array_keys($options)));
}
if ($displaylist) {
$mform->addElement('select', 'newparent', get_string('movecategorycontentto'), $displaylist);
if (in_array($category->parent, $displaylist)) {
$mform->setDefault('newparent', $category->parent);
}
$mform->disabledIf('newparent', 'fulldelete', 'eq', '1');
}
} else {
$mform->addElement('hidden', 'fulldelete', 1);
$mform->addElement('static', 'emptymessage', '', get_string('deletecategoryempty'));
} }
$mform->addElement('hidden', 'delete'); $mform->addElement('hidden', 'delete');
@ -73,12 +126,9 @@ class delete_category_form extends moodleform {
function validation($data, $files) { function validation($data, $files) {
$errors = parent::validation($data, $files); $errors = parent::validation($data, $files);
if (!empty($data['fulldelete'])) { if (empty($data['fulldelete']) && empty($data['newparent'])) {
// already verified /// When they have chosen the move option, they must specify a destination.
} else { $errors['newparent'] = get_string('required');
if (empty($data['newparent'])) {
$errors['newparent'] = get_string('required');
}
} }
if ($data['sure'] != md5(serialize($this->_category))) { if ($data['sure'] != md5(serialize($this->_category))) {

View file

@ -108,11 +108,6 @@
require_once($CFG->libdir . '/questionlib.php'); require_once($CFG->libdir . '/questionlib.php');
print_category_edit_header(); print_category_edit_header();
print_heading($heading); print_heading($heading);
print_box(get_string('deletecategorycheck2'), 'generalbox boxwidthnormal boxaligncenter');
if (question_context_has_any_questions($context)) {
print_box(get_string('deletecoursecategorywithquestions', 'question'),
'generalbox boxwidthnormal boxaligncenter');
}
$mform->display(); $mform->display();
admin_externalpage_print_footer(); admin_externalpage_print_footer();
exit(); exit();

View file

@ -1749,8 +1749,9 @@ function get_child_categories($parentid) {
* *
* @param array $list For output, accumulates an array categoryid => full category path name * @param array $list For output, accumulates an array categoryid => full category path name
* @param array $parents For output, accumulates an array categoryid => list of parent category ids. * @param array $parents For output, accumulates an array categoryid => list of parent category ids.
* @param string $requiredcapability if given, only categories where the current * @param string/array $requiredcapability if given, only categories where the current
* user has this capability will be added to $list. * user has this capability will be added to $list. Can also be an array of capabilities,
* in which case they are all required.
* @param integer $excludeid Omit this category and its children from the lists built. * @param integer $excludeid Omit this category and its children from the lists built.
* @param object $category Build the tree starting at this category - otherwise starts at the top level. * @param object $category Build the tree starting at this category - otherwise starts at the top level.
* @param string $path For internal use, as part of recursive calls. * @param string $path For internal use, as part of recursive calls.
@ -1787,7 +1788,7 @@ function make_categories_list(&$list, &$parents, $requiredcapability = '',
if ($requiredcapability) { if ($requiredcapability) {
ensure_context_subobj_present($category, CONTEXT_COURSECAT); ensure_context_subobj_present($category, CONTEXT_COURSECAT);
} }
if (!$requiredcapability || has_capability($requiredcapability, $category->context)) { if (!$requiredcapability || has_all_capabilities($requiredcapability, $category->context)) {
$list[$category->id] = $path; $list[$category->id] = $path;
} }
} }

View file

@ -467,5 +467,6 @@ $string['wrongroleid'] = 'Incorrect role ID!';
$string['wrongsourcebase'] = 'Wrong source URL base'; $string['wrongsourcebase'] = 'Wrong source URL base';
$string['wrongzipfilename'] = 'Wrong ZIP file name'; $string['wrongzipfilename'] = 'Wrong ZIP file name';
$string['xmldberror'] = 'XMLDB error!'; $string['xmldberror'] = 'XMLDB error!';
$string['youcannotdeletecategory'] = 'You cannot delete category \'$a\' becuase you can neither delete the contents, nor move them elsewhere.';
?> ?>

View file

@ -398,11 +398,13 @@ $string['defaultcourseteacherdescription'] = 'Teachers can do anything within a
$string['defaultcourseteachers'] = 'Teachers'; $string['defaultcourseteachers'] = 'Teachers';
$string['delete'] = 'Delete'; $string['delete'] = 'Delete';
$string['deleteall'] = 'Delete all'; $string['deleteall'] = 'Delete all';
$string['deleteallcannotundo'] = 'Delete all - cannot be undone';
$string['deleteallcomments'] = 'Delete all comments'; $string['deleteallcomments'] = 'Delete all comments';
$string['deleteallratings'] = 'Delete all ratings'; $string['deleteallratings'] = 'Delete all ratings';
$string['deletecategory'] = 'Delete category: $a'; $string['deletecategory'] = 'Delete category: $a';
$string['deletecategorycheck'] = 'Are you absolutely sure you want to completely delete this category <b>\'$a\'</b>?<br />This will move all courses into the parent category if there is one, or into Miscellaneous.'; $string['deletecategorycheck'] = 'Are you absolutely sure you want to completely delete this category <b>\'$a\'</b>?<br />This will move all courses into the parent category if there is one, or into Miscellaneous.';
$string['deletecategorycheck2'] = 'If you delete this category, you need to choose what to do with the courses and subcategories it contains.'; $string['deletecategorycheck2'] = 'If you delete this category, you need to choose what to do with the courses and subcategories it contains.';
$string['deletecategoryempty'] = 'This category is empty.';
$string['deletecheck'] = 'Delete $a ?'; $string['deletecheck'] = 'Delete $a ?';
$string['deletecheckfiles'] = 'Are you absolutely sure you want to delete these files?'; $string['deletecheckfiles'] = 'Are you absolutely sure you want to delete these files?';
$string['deletecheckfull'] = 'Are you absolutely sure you want to completely delete $a ?'; $string['deletecheckfull'] = 'Are you absolutely sure you want to completely delete $a ?';
@ -1013,6 +1015,7 @@ $string['mostrecently'] = 'most recently';
$string['move'] = 'Move'; $string['move'] = 'Move';
$string['movecategoryto'] = 'Move category to:'; $string['movecategoryto'] = 'Move category to:';
$string['movecategorycontentto'] = 'Move into'; $string['movecategorycontentto'] = 'Move into';
$string['movecontentstoanothercategory'] = 'Move contents to another category';
$string['movecourseto'] = 'Move course to:'; $string['movecourseto'] = 'Move course to:';
$string['movedown'] = 'Move down'; $string['movedown'] = 'Move down';
$string['movefilestohere'] = 'Move files to here'; $string['movefilestohere'] = 'Move files to here';
@ -1259,6 +1262,7 @@ $string['publicdirectory2'] = 'Publish the site name with a link';
$string['publicdirectorytitle'] = 'See the current list of sites'; $string['publicdirectorytitle'] = 'See the current list of sites';
$string['publicsitefileswarning'] = 'Note: files placed here can be accessed by anyone'; $string['publicsitefileswarning'] = 'Note: files placed here can be accessed by anyone';
$string['question'] = 'Question'; $string['question'] = 'Question';
$string['questionsinthequestionbank'] = 'Questions in the question bank';
$string['readinginfofrombackup'] = 'Reading info from backup'; $string['readinginfofrombackup'] = 'Reading info from backup';
$string['readme'] = 'README'; $string['readme'] = 'README';
$string['recentactivity'] = 'Recent Activity'; $string['recentactivity'] = 'Recent Activity';
@ -1535,6 +1539,7 @@ $string['theme'] = 'Theme';
$string['themes'] = 'Themes'; $string['themes'] = 'Themes';
$string['themesaved'] = 'New theme saved'; $string['themesaved'] = 'New theme saved';
$string['thereareno'] = 'There are no $a in this course'; $string['thereareno'] = 'There are no $a in this course';
$string['thiscategorycontains'] = 'This category contains';
$string['thischarset'] = 'UTF-8'; $string['thischarset'] = 'UTF-8';
$string['thisdirection'] = 'ltr'; $string['thisdirection'] = 'ltr';
$string['thislanguage'] = 'English'; $string['thislanguage'] = 'English';
@ -1680,6 +1685,7 @@ within the course so that we can learn more about you:
$a->profileurl'; $a->profileurl';
$string['whattocallzip'] = 'What do you want to call the zip file?'; $string['whattocallzip'] = 'What do you want to call the zip file?';
$string['whattodo'] = 'What to do';
$string['withchosenfiles'] = 'With chosen files'; $string['withchosenfiles'] = 'With chosen files';
$string['withoutuserdata'] = 'without user data'; $string['withoutuserdata'] = 'without user data';
$string['withselectedusers'] = 'With selected users...'; $string['withselectedusers'] = 'With selected users...';

View file

@ -37,6 +37,8 @@
* *
* Whether the user can do something... * Whether the user can do something...
* - has_capability() * - has_capability()
* - has_any_capability()
* - has_all_capabilities()
* - require_capability() * - require_capability()
* - require_login() (from moodlelib) * - require_login() (from moodlelib)
* *
@ -496,6 +498,28 @@ function has_any_capability($capabilities, $context, $userid=NULL, $doanything=t
return false; return false;
} }
/**
* This function returns whether the current user has all of the capabilities in the
* $capabilities array. This is a simple wrapper around has_capability for convinience.
*
* There are probably tricks that could be done to improve the performance here, for example,
* check the capabilities that are already cached first.
*
* @param array $capabilities - an array of capability names.
* @param object $context - a context object (record from context table)
* @param integer $userid - a userid number, empty if current $USER
* @param bool $doanything - if false, ignore do anything
* @return bool
*/
function has_all_capabilities($capabilities, $context, $userid=NULL, $doanything=true) {
foreach ($capabilities as $capability) {
if (!has_capability($capability, $context, $userid, $doanything)) {
return false;
}
}
return true;
}
/** /**
* Uses 1 DB query to answer whether a user is an admin at the sitelevel. * Uses 1 DB query to answer whether a user is an admin at the sitelevel.
* It depends on DB schema >=1.7 but does not depend on the new datastructures * It depends on DB schema >=1.7 but does not depend on the new datastructures