mirror of
https://github.com/moodle/moodle.git
synced 2025-01-19 14:27:22 +01:00
fc6a0fe294
Newly introduced unit tests are failing because the timestamps used are too large. Example timestamps were 11 characters long instead of 10.
734 lines
38 KiB
PHP
734 lines
38 KiB
PHP
<?php
|
||
// This file is part of Moodle - http://moodle.org/
|
||
//
|
||
// 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.
|
||
//
|
||
// You should have received a copy of the GNU General Public License
|
||
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
|
||
|
||
/**
|
||
* Tests for class coursecat from lib/coursecatlib.php
|
||
*
|
||
* @package core
|
||
* @category phpunit
|
||
* @copyright 2013 Marina Glancy
|
||
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
||
*/
|
||
|
||
defined('MOODLE_INTERNAL') || die();
|
||
|
||
global $CFG;
|
||
require_once($CFG->libdir . '/coursecatlib.php');
|
||
|
||
/**
|
||
* Functional test for coursecatlib.php
|
||
*/
|
||
class core_coursecatlib_testcase extends advanced_testcase {
|
||
|
||
protected $roles;
|
||
|
||
protected function setUp() {
|
||
parent::setUp();
|
||
$this->resetAfterTest();
|
||
$user = $this->getDataGenerator()->create_user();
|
||
$this->setUser($user);
|
||
}
|
||
|
||
protected function get_roleid($context = null) {
|
||
global $USER;
|
||
if ($context === null) {
|
||
$context = context_system::instance();
|
||
}
|
||
if (is_object($context)) {
|
||
$context = $context->id;
|
||
}
|
||
if (empty($this->roles)) {
|
||
$this->roles = array();
|
||
}
|
||
if (empty($this->roles[$USER->id])) {
|
||
$this->roles[$USER->id] = array();
|
||
}
|
||
if (empty($this->roles[$USER->id][$context])) {
|
||
$this->roles[$USER->id][$context] = create_role('Role for '.$USER->id.' in '.$context, 'role'.$USER->id.'-'.$context, '-');
|
||
role_assign($this->roles[$USER->id][$context], $USER->id, $context);
|
||
}
|
||
return $this->roles[$USER->id][$context];
|
||
}
|
||
|
||
protected function assign_capability($capability, $permission = CAP_ALLOW, $contextid = null) {
|
||
if ($contextid === null) {
|
||
$contextid = context_system::instance();
|
||
}
|
||
if (is_object($contextid)) {
|
||
$contextid = $contextid->id;
|
||
}
|
||
assign_capability($capability, $permission, $this->get_roleid($contextid), $contextid, true);
|
||
accesslib_clear_all_caches_for_unit_testing();
|
||
}
|
||
|
||
public function test_create_coursecat() {
|
||
// Create the category.
|
||
$data = new stdClass();
|
||
$data->name = 'aaa';
|
||
$data->description = 'aaa';
|
||
$data->idnumber = '';
|
||
|
||
$category1 = coursecat::create($data);
|
||
|
||
// Initially confirm that base data was inserted correctly.
|
||
$this->assertSame($data->name, $category1->name);
|
||
$this->assertSame($data->description, $category1->description);
|
||
$this->assertSame($data->idnumber, $category1->idnumber);
|
||
|
||
$this->assertGreaterThanOrEqual(1, $category1->sortorder);
|
||
|
||
// Create two more categories and test the sortorder worked correctly.
|
||
$data->name = 'ccc';
|
||
$category2 = coursecat::create($data);
|
||
|
||
$data->name = 'bbb';
|
||
$category3 = coursecat::create($data);
|
||
|
||
$this->assertGreaterThan($category1->sortorder, $category2->sortorder);
|
||
$this->assertGreaterThan($category2->sortorder, $category3->sortorder);
|
||
}
|
||
|
||
public function test_name_idnumber_exceptions() {
|
||
try {
|
||
coursecat::create(array('name' => ''));
|
||
$this->fail('Missing category name exception expected in coursecat::create');
|
||
} catch (moodle_exception $e) {
|
||
$this->assertInstanceOf('moodle_exception', $e);
|
||
}
|
||
$cat1 = coursecat::create(array('name' => 'Cat1', 'idnumber' => '1'));
|
||
try {
|
||
$cat1->update(array('name' => ''));
|
||
$this->fail('Missing category name exception expected in coursecat::update');
|
||
} catch (moodle_exception $e) {
|
||
$this->assertInstanceOf('moodle_exception', $e);
|
||
}
|
||
try {
|
||
coursecat::create(array('name' => 'Cat2', 'idnumber' => '1'));
|
||
$this->fail('Duplicate idnumber exception expected in coursecat::create');
|
||
} catch (moodle_exception $e) {
|
||
$this->assertInstanceOf('moodle_exception', $e);
|
||
}
|
||
$cat2 = coursecat::create(array('name' => 'Cat2', 'idnumber' => '2'));
|
||
try {
|
||
$cat2->update(array('idnumber' => '1'));
|
||
$this->fail('Duplicate idnumber exception expected in coursecat::update');
|
||
} catch (moodle_exception $e) {
|
||
$this->assertInstanceOf('moodle_exception', $e);
|
||
}
|
||
}
|
||
|
||
public function test_visibility() {
|
||
$this->assign_capability('moodle/category:viewhiddencategories');
|
||
$this->assign_capability('moodle/category:manage');
|
||
|
||
// Create category 1 initially hidden.
|
||
$category1 = coursecat::create(array('name' => 'Cat1', 'visible' => 0));
|
||
$this->assertEquals(0, $category1->visible);
|
||
$this->assertEquals(0, $category1->visibleold);
|
||
|
||
// Create category 2 initially hidden as a child of hidden category 1.
|
||
$category2 = coursecat::create(array('name' => 'Cat2', 'visible' => 0, 'parent' => $category1->id));
|
||
$this->assertEquals(0, $category2->visible);
|
||
$this->assertEquals(0, $category2->visibleold);
|
||
|
||
// Create category 3 initially visible as a child of hidden category 1.
|
||
$category3 = coursecat::create(array('name' => 'Cat3', 'visible' => 1, 'parent' => $category1->id));
|
||
$this->assertEquals(0, $category3->visible);
|
||
$this->assertEquals(1, $category3->visibleold);
|
||
|
||
// Show category 1 and make sure that category 2 is hidden and category 3 is visible.
|
||
$category1->show();
|
||
$this->assertEquals(1, coursecat::get($category1->id)->visible);
|
||
$this->assertEquals(0, coursecat::get($category2->id)->visible);
|
||
$this->assertEquals(1, coursecat::get($category3->id)->visible);
|
||
|
||
// Create visible category 4.
|
||
$category4 = coursecat::create(array('name' => 'Cat4'));
|
||
$this->assertEquals(1, $category4->visible);
|
||
$this->assertEquals(1, $category4->visibleold);
|
||
|
||
// Create visible category 5 as a child of visible category 4.
|
||
$category5 = coursecat::create(array('name' => 'Cat5', 'parent' => $category4->id));
|
||
$this->assertEquals(1, $category5->visible);
|
||
$this->assertEquals(1, $category5->visibleold);
|
||
|
||
// Hide category 4 and make sure category 5 is hidden too.
|
||
$category4->hide();
|
||
$this->assertEquals(0, $category4->visible);
|
||
$this->assertEquals(0, $category4->visibleold);
|
||
$category5 = coursecat::get($category5->id); // We have to re-read from DB.
|
||
$this->assertEquals(0, $category5->visible);
|
||
$this->assertEquals(1, $category5->visibleold);
|
||
|
||
// Show category 4 and make sure category 5 is visible too.
|
||
$category4->show();
|
||
$this->assertEquals(1, $category4->visible);
|
||
$this->assertEquals(1, $category4->visibleold);
|
||
$category5 = coursecat::get($category5->id); // We have to re-read from DB.
|
||
$this->assertEquals(1, $category5->visible);
|
||
$this->assertEquals(1, $category5->visibleold);
|
||
|
||
// Move category 5 under hidden category 2 and make sure it became hidden.
|
||
$category5->change_parent($category2->id);
|
||
$this->assertEquals(0, $category5->visible);
|
||
$this->assertEquals(1, $category5->visibleold);
|
||
|
||
// Re-read object for category 5 from DB and check again.
|
||
$category5 = coursecat::get($category5->id);
|
||
$this->assertEquals(0, $category5->visible);
|
||
$this->assertEquals(1, $category5->visibleold);
|
||
|
||
// Rricky one! Move hidden category 5 under visible category ("Top") and make sure it is still hidden-
|
||
// WHY? Well, different people may expect different behaviour here. So better keep it hidden.
|
||
$category5->change_parent(0);
|
||
$this->assertEquals(0, $category5->visible);
|
||
$this->assertEquals(1, $category5->visibleold);
|
||
}
|
||
|
||
public function test_hierarchy() {
|
||
$this->assign_capability('moodle/category:viewhiddencategories');
|
||
$this->assign_capability('moodle/category:manage');
|
||
|
||
$category1 = coursecat::create(array('name' => 'Cat1'));
|
||
$category2 = coursecat::create(array('name' => 'Cat2', 'parent' => $category1->id));
|
||
$category3 = coursecat::create(array('name' => 'Cat3', 'parent' => $category1->id));
|
||
$category4 = coursecat::create(array('name' => 'Cat4', 'parent' => $category2->id));
|
||
|
||
// Check function get_children().
|
||
$this->assertEquals(array($category2->id, $category3->id), array_keys($category1->get_children()));
|
||
// Check function get_parents().
|
||
$this->assertEquals(array($category1->id, $category2->id), $category4->get_parents());
|
||
|
||
// Can not move category to itself or to it's children.
|
||
$this->assertFalse($category1->can_change_parent($category2->id));
|
||
$this->assertFalse($category2->can_change_parent($category2->id));
|
||
// Can move category to grandparent.
|
||
$this->assertTrue($category4->can_change_parent($category1->id));
|
||
|
||
try {
|
||
$category2->change_parent($category4->id);
|
||
$this->fail('Exception expected - can not move category');
|
||
} catch (moodle_exception $e) {
|
||
$this->assertInstanceOf('moodle_exception', $e);
|
||
}
|
||
|
||
$category4->change_parent(0);
|
||
$this->assertEquals(array(), $category4->get_parents());
|
||
$this->assertEquals(array($category2->id, $category3->id), array_keys($category1->get_children()));
|
||
$this->assertEquals(array(), array_keys($category2->get_children()));
|
||
}
|
||
|
||
public function test_update() {
|
||
$category1 = coursecat::create(array('name' => 'Cat1'));
|
||
$timecreated = $category1->timemodified;
|
||
$this->assertSame('Cat1', $category1->name);
|
||
$this->assertTrue(empty($category1->description));
|
||
sleep(2);
|
||
$testdescription = 'This is cat 1 а также русский текст';
|
||
$category1->update(array('description' => $testdescription));
|
||
$this->assertSame($testdescription, $category1->description);
|
||
$category1 = coursecat::get($category1->id);
|
||
$this->assertSame($testdescription, $category1->description);
|
||
cache_helper::purge_by_event('changesincoursecat');
|
||
$category1 = coursecat::get($category1->id);
|
||
$this->assertSame($testdescription, $category1->description);
|
||
|
||
$this->assertGreaterThan($timecreated, $category1->timemodified);
|
||
}
|
||
|
||
public function test_delete() {
|
||
global $DB;
|
||
|
||
$this->assign_capability('moodle/category:manage');
|
||
$this->assign_capability('moodle/course:create');
|
||
|
||
$initialcatid = $DB->get_field_sql('SELECT max(id) from {course_categories}');
|
||
|
||
$category1 = coursecat::create(array('name' => 'Cat1'));
|
||
$category2 = coursecat::create(array('name' => 'Cat2', 'parent' => $category1->id));
|
||
$category3 = coursecat::create(array('name' => 'Cat3'));
|
||
$category4 = coursecat::create(array('name' => 'Cat4', 'parent' => $category2->id));
|
||
|
||
$course1 = $this->getDataGenerator()->create_course(array('category' => $category2->id));
|
||
$course2 = $this->getDataGenerator()->create_course(array('category' => $category4->id));
|
||
$course3 = $this->getDataGenerator()->create_course(array('category' => $category4->id));
|
||
$course4 = $this->getDataGenerator()->create_course(array('category' => $category1->id));
|
||
|
||
// Now we have
|
||
// $category1
|
||
// $category2
|
||
// $category4
|
||
// $course2
|
||
// $course3
|
||
// $course1
|
||
// $course4
|
||
// $category3
|
||
// structure.
|
||
|
||
// Login as another user to test course:delete capability (user who created course can delete it within 24h even without cap).
|
||
$this->setUser($this->getDataGenerator()->create_user());
|
||
|
||
// Delete category 2 and move content to category 3.
|
||
$this->assertFalse($category2->can_move_content_to($category3->id)); // No luck!
|
||
// Add necessary capabilities.
|
||
$this->assign_capability('moodle/course:create', CAP_ALLOW, context_coursecat::instance($category3->id));
|
||
$this->assign_capability('moodle/category:manage');
|
||
$this->assertTrue($category2->can_move_content_to($category3->id)); // Hurray!
|
||
$category2->delete_move($category3->id);
|
||
|
||
// Make sure we have:
|
||
// $category1
|
||
// $course4
|
||
// $category3
|
||
// $category4
|
||
// $course2
|
||
// $course3
|
||
// $course1
|
||
// structure.
|
||
|
||
$this->assertNull(coursecat::get($category2->id, IGNORE_MISSING, true));
|
||
$this->assertEquals(array(), $category1->get_children());
|
||
$this->assertEquals(array($category4->id), array_keys($category3->get_children()));
|
||
$this->assertEquals($category4->id, $DB->get_field('course', 'category', array('id' => $course2->id)));
|
||
$this->assertEquals($category4->id, $DB->get_field('course', 'category', array('id' => $course3->id)));
|
||
$this->assertEquals($category3->id, $DB->get_field('course', 'category', array('id' => $course1->id)));
|
||
|
||
// Delete category 3 completely.
|
||
$this->assertFalse($category3->can_delete_full()); // No luck!
|
||
// Add necessary capabilities.
|
||
$this->assign_capability('moodle/course:delete', CAP_ALLOW, context_coursecat::instance($category3->id));
|
||
$this->assertTrue($category3->can_delete_full()); // Hurray!
|
||
$category3->delete_full();
|
||
|
||
// Make sure we have:
|
||
// $category1
|
||
// $course4
|
||
// structure.
|
||
|
||
// Note that we also have default 'Miscellaneous' category and default 'site' course.
|
||
$this->assertEquals(1, $DB->get_field_sql('SELECT count(*) FROM {course_categories} WHERE id > ?', array($initialcatid)));
|
||
$this->assertEquals($category1->id, $DB->get_field_sql('SELECT max(id) FROM {course_categories}'));
|
||
$this->assertEquals(1, $DB->get_field_sql('SELECT count(*) FROM {course} WHERE id <> ?', array(SITEID)));
|
||
$this->assertEquals(array('id' => $course4->id, 'category' => $category1->id),
|
||
(array)$DB->get_record_sql('SELECT id, category from {course} where id <> ?', array(SITEID)));
|
||
}
|
||
|
||
public function test_get_children() {
|
||
$category1 = coursecat::create(array('name' => 'Cat1'));
|
||
$category2 = coursecat::create(array('name' => 'Cat2', 'parent' => $category1->id));
|
||
$category3 = coursecat::create(array('name' => 'Cat3', 'parent' => $category1->id, 'visible' => 0));
|
||
$category4 = coursecat::create(array('name' => 'Cat4', 'idnumber' => '12', 'parent' => $category1->id));
|
||
$category5 = coursecat::create(array('name' => 'Cat5', 'idnumber' => '11', 'parent' => $category1->id, 'visible' => 0));
|
||
$category6 = coursecat::create(array('name' => 'Cat6', 'idnumber' => '10', 'parent' => $category1->id));
|
||
$category7 = coursecat::create(array('name' => 'Cat0', 'parent' => $category1->id));
|
||
|
||
$children = $category1->get_children();
|
||
// User does not have the capability to view hidden categories, so the list should be
|
||
// 2, 4, 6, 7.
|
||
$this->assertEquals(array($category2->id, $category4->id, $category6->id, $category7->id), array_keys($children));
|
||
$this->assertEquals(4, $category1->get_children_count());
|
||
|
||
$children = $category1->get_children(array('offset' => 2));
|
||
$this->assertEquals(array($category6->id, $category7->id), array_keys($children));
|
||
$this->assertEquals(4, $category1->get_children_count());
|
||
|
||
$children = $category1->get_children(array('limit' => 2));
|
||
$this->assertEquals(array($category2->id, $category4->id), array_keys($children));
|
||
|
||
$children = $category1->get_children(array('offset' => 1, 'limit' => 2));
|
||
$this->assertEquals(array($category4->id, $category6->id), array_keys($children));
|
||
|
||
$children = $category1->get_children(array('sort' => array('name' => 1)));
|
||
// Must be 7, 2, 4, 6.
|
||
$this->assertEquals(array($category7->id, $category2->id, $category4->id, $category6->id), array_keys($children));
|
||
|
||
$children = $category1->get_children(array('sort' => array('idnumber' => 1, 'name' => -1)));
|
||
// Must be 2, 7, 6, 4.
|
||
$this->assertEquals(array($category2->id, $category7->id, $category6->id, $category4->id), array_keys($children));
|
||
|
||
// Check that everything is all right after purging the caches.
|
||
cache_helper::purge_by_event('changesincoursecat');
|
||
$children = $category1->get_children();
|
||
$this->assertEquals(array($category2->id, $category4->id, $category6->id, $category7->id), array_keys($children));
|
||
$this->assertEquals(4, $category1->get_children_count());
|
||
}
|
||
|
||
/**
|
||
* Test the countall function
|
||
*/
|
||
public function test_count_all() {
|
||
global $DB;
|
||
// Dont assume there is just one. An add-on might create a category as part of the install.
|
||
$numcategories = $DB->count_records('course_categories');
|
||
$this->assertEquals($numcategories, coursecat::count_all());
|
||
$category1 = coursecat::create(array('name' => 'Cat1'));
|
||
$category2 = coursecat::create(array('name' => 'Cat2', 'parent' => $category1->id));
|
||
$category3 = coursecat::create(array('name' => 'Cat3', 'parent' => $category2->id, 'visible' => 0));
|
||
// Now we've got three more.
|
||
$this->assertEquals($numcategories + 3, coursecat::count_all());
|
||
cache_helper::purge_by_event('changesincoursecat');
|
||
// We should still have 4.
|
||
$this->assertEquals($numcategories + 3, coursecat::count_all());
|
||
}
|
||
|
||
/**
|
||
* Test a categories ability to resort courses.
|
||
*/
|
||
public function test_resort_courses() {
|
||
$this->resetAfterTest(true);
|
||
$generator = $this->getDataGenerator();
|
||
$category = $generator->create_category();
|
||
$course1 = $generator->create_course(array(
|
||
'category' => $category->id,
|
||
'idnumber' => '006-01',
|
||
'shortname' => 'Biome Study',
|
||
'fullname' => '<span lang="ar" class="multilang">'.'دراسة منطقة إحيائية'.'</span><span lang="en" class="multilang">Biome Study</span>',
|
||
'timecreated' => '1000000001'
|
||
));
|
||
$course2 = $generator->create_course(array(
|
||
'category' => $category->id,
|
||
'idnumber' => '007-02',
|
||
'shortname' => 'Chemistry Revision',
|
||
'fullname' => 'Chemistry Revision',
|
||
'timecreated' => '1000000002'
|
||
));
|
||
$course3 = $generator->create_course(array(
|
||
'category' => $category->id,
|
||
'idnumber' => '007-03',
|
||
'shortname' => 'Swiss Rolls and Sunflowers',
|
||
'fullname' => 'Aarkvarks guide to Swiss Rolls and Sunflowers',
|
||
'timecreated' => '1000000003'
|
||
));
|
||
$course4 = $generator->create_course(array(
|
||
'category' => $category->id,
|
||
'idnumber' => '006-04',
|
||
'shortname' => 'Scratch',
|
||
'fullname' => '<a href="test.php">Basic Scratch</a>',
|
||
'timecreated' => '1000000004'
|
||
));
|
||
$c1 = (int)$course1->id;
|
||
$c2 = (int)$course2->id;
|
||
$c3 = (int)$course3->id;
|
||
$c4 = (int)$course4->id;
|
||
|
||
$coursecat = coursecat::get($category->id);
|
||
$this->assertTrue($coursecat->resort_courses('idnumber'));
|
||
$this->assertSame(array($c1, $c4, $c2, $c3), array_keys($coursecat->get_courses()));
|
||
|
||
$this->assertTrue($coursecat->resort_courses('shortname'));
|
||
$this->assertSame(array($c1, $c2, $c4, $c3), array_keys($coursecat->get_courses()));
|
||
|
||
$this->assertTrue($coursecat->resort_courses('timecreated'));
|
||
$this->assertSame(array($c1, $c2, $c3, $c4), array_keys($coursecat->get_courses()));
|
||
|
||
try {
|
||
// Enable the multilang filter and set it to apply to headings and content.
|
||
filter_set_global_state('multilang', TEXTFILTER_ON);
|
||
filter_set_applies_to_strings('multilang', true);
|
||
$expected = array($c3, $c4, $c1, $c2);
|
||
} catch (coding_exception $ex) {
|
||
$expected = array($c3, $c4, $c2, $c1);
|
||
}
|
||
$this->assertTrue($coursecat->resort_courses('fullname'));
|
||
$this->assertSame($expected, array_keys($coursecat->get_courses()));
|
||
}
|
||
|
||
public function test_get_search_courses() {
|
||
$cat1 = coursecat::create(array('name' => 'Cat1'));
|
||
$cat2 = coursecat::create(array('name' => 'Cat2', 'parent' => $cat1->id));
|
||
$c1 = $this->getDataGenerator()->create_course(array('category' => $cat1->id, 'fullname' => 'Test 3', 'summary' => ' ', 'idnumber' => 'ID3'));
|
||
$c2 = $this->getDataGenerator()->create_course(array('category' => $cat1->id, 'fullname' => 'Test 1', 'summary' => ' ', 'visible' => 0));
|
||
$c3 = $this->getDataGenerator()->create_course(array('category' => $cat1->id, 'fullname' => 'Математика', 'summary' => ' Test '));
|
||
$c4 = $this->getDataGenerator()->create_course(array('category' => $cat1->id, 'fullname' => 'Test 4', 'summary' => ' ', 'idnumber' => 'ID4'));
|
||
|
||
$c5 = $this->getDataGenerator()->create_course(array('category' => $cat2->id, 'fullname' => 'Test 5', 'summary' => ' '));
|
||
$c6 = $this->getDataGenerator()->create_course(array('category' => $cat2->id, 'fullname' => 'Дискретная Математика', 'summary' => ' '));
|
||
$c7 = $this->getDataGenerator()->create_course(array('category' => $cat2->id, 'fullname' => 'Test 7', 'summary' => ' ', 'visible' => 0));
|
||
$c8 = $this->getDataGenerator()->create_course(array('category' => $cat2->id, 'fullname' => 'Test 8', 'summary' => ' '));
|
||
|
||
// Get courses in category 1 (returned visible only because user is not enrolled).
|
||
$res = $cat1->get_courses(array('sortorder' => 1));
|
||
$this->assertEquals(array($c4->id, $c3->id, $c1->id), array_keys($res)); // Courses are added in reverse order.
|
||
$this->assertEquals(3, $cat1->get_courses_count());
|
||
|
||
// Get courses in category 1 recursively (returned visible only because user is not enrolled).
|
||
$res = $cat1->get_courses(array('recursive' => 1));
|
||
$this->assertEquals(array($c4->id, $c3->id, $c1->id, $c8->id, $c6->id, $c5->id), array_keys($res));
|
||
$this->assertEquals(6, $cat1->get_courses_count(array('recursive' => 1)));
|
||
|
||
// Get courses sorted by fullname.
|
||
$res = $cat1->get_courses(array('sort' => array('fullname' => 1)));
|
||
$this->assertEquals(array($c1->id, $c4->id, $c3->id), array_keys($res));
|
||
$this->assertEquals(3, $cat1->get_courses_count(array('sort' => array('fullname' => 1))));
|
||
|
||
// Get courses sorted by fullname recursively.
|
||
$res = $cat1->get_courses(array('recursive' => 1, 'sort' => array('fullname' => 1)));
|
||
$this->assertEquals(array($c1->id, $c4->id, $c5->id, $c8->id, $c6->id, $c3->id), array_keys($res));
|
||
$this->assertEquals(6, $cat1->get_courses_count(array('recursive' => 1, 'sort' => array('fullname' => 1))));
|
||
|
||
// Get courses sorted by fullname recursively, use offset and limit.
|
||
$res = $cat1->get_courses(array('recursive' => 1, 'offset' => 1, 'limit' => 2, 'sort' => array('fullname' => -1)));
|
||
$this->assertEquals(array($c6->id, $c8->id), array_keys($res));
|
||
// Offset and limit do not affect get_courses_count().
|
||
$this->assertEquals(6, $cat1->get_courses_count(array('recursive' => 1, 'offset' => 1, 'limit' => 2, 'sort' => array('fullname' => 1))));
|
||
|
||
// Calling get_courses_count without prior call to get_courses().
|
||
$this->assertEquals(3, $cat2->get_courses_count(array('recursive' => 1, 'sort' => array('idnumber' => 1))));
|
||
|
||
// Search courses.
|
||
|
||
// Search by text.
|
||
$res = coursecat::search_courses(array('search' => 'Test'));
|
||
$this->assertEquals(array($c4->id, $c3->id, $c1->id, $c8->id, $c5->id), array_keys($res));
|
||
$this->assertEquals(5, coursecat::search_courses_count(array('search' => 'Test')));
|
||
|
||
// Search by text with specified offset and limit.
|
||
$options = array('sort' => array('fullname' => 1), 'offset' => 1, 'limit' => 2);
|
||
$res = coursecat::search_courses(array('search' => 'Test'), $options);
|
||
$this->assertEquals(array($c4->id, $c5->id), array_keys($res));
|
||
$this->assertEquals(5, coursecat::search_courses_count(array('search' => 'Test'), $options));
|
||
|
||
// IMPORTANT: the tests below may fail on some databases
|
||
// case-insensitive search.
|
||
$res = coursecat::search_courses(array('search' => 'test'));
|
||
$this->assertEquals(array($c4->id, $c3->id, $c1->id, $c8->id, $c5->id), array_keys($res));
|
||
$this->assertEquals(5, coursecat::search_courses_count(array('search' => 'test')));
|
||
|
||
// Non-latin language search.
|
||
$res = coursecat::search_courses(array('search' => 'Математика'));
|
||
$this->assertEquals(array($c3->id, $c6->id), array_keys($res));
|
||
$this->assertEquals(2, coursecat::search_courses_count(array('search' => 'Математика'), array()));
|
||
}
|
||
|
||
public function test_course_contacts() {
|
||
global $DB, $CFG;
|
||
$teacherrole = $DB->get_record('role', array('shortname'=>'editingteacher'));
|
||
$managerrole = $DB->get_record('role', array('shortname'=>'manager'));
|
||
$studentrole = $DB->get_record('role', array('shortname'=>'student'));
|
||
$oldcoursecontact = $CFG->coursecontact;
|
||
|
||
$CFG->coursecontact = $managerrole->id. ','. $teacherrole->id;
|
||
|
||
/*
|
||
* User is listed in course contacts for the course if he has one of the
|
||
* "course contact" roles ($CFG->coursecontact) AND is enrolled in the course.
|
||
* If the user has several roles only the highest is displayed.
|
||
*/
|
||
|
||
// Test case:
|
||
//
|
||
// == Cat1 (user2 has teacher role)
|
||
// == Cat2
|
||
// -- course21 (user2 is enrolled as manager) | [Expected] Manager: F2 L2
|
||
// -- course22 (user2 is enrolled as student) | [Expected] Teacher: F2 L2
|
||
// == Cat4 (user2 has manager role)
|
||
// -- course41 (user4 is enrolled as teacher, user5 is enrolled as manager) | [Expected] Manager: F5 L5, Teacher: F4 L4
|
||
// -- course42 (user2 is enrolled as teacher) | [Expected] Manager: F2 L2
|
||
// == Cat3 (user3 has manager role)
|
||
// -- course31 (user3 is enrolled as student) | [Expected] Manager: F3 L3
|
||
// -- course32 | [Expected]
|
||
// -- course11 (user1 is enrolled as teacher) | [Expected] Teacher: F1 L1
|
||
// -- course12 (user1 has teacher role) | [Expected]
|
||
// also user4 is enrolled as teacher but enrolment is not active
|
||
$category = $course = $enrol = $user = array();
|
||
$category[1] = coursecat::create(array('name' => 'Cat1'))->id;
|
||
$category[2] = coursecat::create(array('name' => 'Cat2', 'parent' => $category[1]))->id;
|
||
$category[3] = coursecat::create(array('name' => 'Cat3', 'parent' => $category[1]))->id;
|
||
$category[4] = coursecat::create(array('name' => 'Cat4', 'parent' => $category[2]))->id;
|
||
foreach (array(1, 2, 3, 4) as $catid) {
|
||
foreach (array(1, 2) as $courseid) {
|
||
$course[$catid][$courseid] = $this->getDataGenerator()->create_course(array('idnumber' => 'id'.$catid.$courseid,
|
||
'category' => $category[$catid]))->id;
|
||
$enrol[$catid][$courseid] = $DB->get_record('enrol', array('courseid'=>$course[$catid][$courseid], 'enrol'=>'manual'), '*', MUST_EXIST);
|
||
}
|
||
}
|
||
foreach (array(1, 2, 3, 4, 5) as $userid) {
|
||
$user[$userid] = $this->getDataGenerator()->create_user(array('firstname' => 'F'.$userid, 'lastname' => 'L'.$userid))->id;
|
||
}
|
||
|
||
$manual = enrol_get_plugin('manual');
|
||
|
||
// Cat1 (user2 has teacher role)
|
||
role_assign($teacherrole->id, $user[2], context_coursecat::instance($category[1]));
|
||
// course21 (user2 is enrolled as manager)
|
||
$manual->enrol_user($enrol[2][1], $user[2], $managerrole->id);
|
||
// course22 (user2 is enrolled as student)
|
||
$manual->enrol_user($enrol[2][2], $user[2], $studentrole->id);
|
||
// Cat4 (user2 has manager role)
|
||
role_assign($managerrole->id, $user[2], context_coursecat::instance($category[4]));
|
||
// course41 (user4 is enrolled as teacher, user5 is enrolled as manager)
|
||
$manual->enrol_user($enrol[4][1], $user[4], $teacherrole->id);
|
||
$manual->enrol_user($enrol[4][1], $user[5], $managerrole->id);
|
||
// course42 (user2 is enrolled as teacher)
|
||
$manual->enrol_user($enrol[4][2], $user[2], $teacherrole->id);
|
||
// Cat3 (user3 has manager role)
|
||
role_assign($managerrole->id, $user[3], context_coursecat::instance($category[3]));
|
||
// course31 (user3 is enrolled as student)
|
||
$manual->enrol_user($enrol[3][1], $user[3], $studentrole->id);
|
||
// course11 (user1 is enrolled as teacher)
|
||
$manual->enrol_user($enrol[1][1], $user[1], $teacherrole->id);
|
||
// -- course12 (user1 has teacher role)
|
||
// also user4 is enrolled as teacher but enrolment is not active
|
||
role_assign($teacherrole->id, $user[1], context_course::instance($course[1][2]));
|
||
$manual->enrol_user($enrol[1][2], $user[4], $teacherrole->id, 0, 0, ENROL_USER_SUSPENDED);
|
||
|
||
$allcourses = coursecat::get(0)->get_courses(array('recursive' => true, 'coursecontacts' => true, 'sort' => array('idnumber' => 1)));
|
||
// Simplify the list of contacts for each course (similar as renderer would do).
|
||
$contacts = array();
|
||
foreach (array(1, 2, 3, 4) as $catid) {
|
||
foreach (array(1, 2) as $courseid) {
|
||
$tmp = array();
|
||
foreach ($allcourses[$course[$catid][$courseid]]->get_course_contacts() as $contact) {
|
||
$tmp[] = $contact['rolename']. ': '. $contact['username'];
|
||
}
|
||
$contacts[$catid][$courseid] = join(', ', $tmp);
|
||
}
|
||
}
|
||
|
||
// Assert:
|
||
// -- course21 (user2 is enrolled as manager) | Manager: F2 L2
|
||
$this->assertSame('Manager: F2 L2', $contacts[2][1]);
|
||
// -- course22 (user2 is enrolled as student) | Teacher: F2 L2
|
||
$this->assertSame('Teacher: F2 L2', $contacts[2][2]);
|
||
// -- course41 (user4 is enrolled as teacher, user5 is enrolled as manager) | Manager: F5 L5, Teacher: F4 L4
|
||
$this->assertSame('Manager: F5 L5, Teacher: F4 L4', $contacts[4][1]);
|
||
// -- course42 (user2 is enrolled as teacher) | [Expected] Manager: F2 L2
|
||
$this->assertSame('Manager: F2 L2', $contacts[4][2]);
|
||
// -- course31 (user3 is enrolled as student) | Manager: F3 L3
|
||
$this->assertSame('Manager: F3 L3', $contacts[3][1]);
|
||
// -- course32 |
|
||
$this->assertSame('', $contacts[3][2]);
|
||
// -- course11 (user1 is enrolled as teacher) | Teacher: F1 L1
|
||
$this->assertSame('Teacher: F1 L1', $contacts[1][1]);
|
||
// -- course12 (user1 has teacher role) |
|
||
$this->assertSame('', $contacts[1][2]);
|
||
|
||
$CFG->coursecontact = $oldcoursecontact;
|
||
}
|
||
|
||
public function test_overview_files() {
|
||
global $CFG;
|
||
$this->setAdminUser();
|
||
$cat1 = coursecat::create(array('name' => 'Cat1'));
|
||
|
||
// Create course c1 with one image file.
|
||
$dratid1 = $this->fill_draft_area(array('filename.jpg' => 'Test file contents1'));
|
||
$c1 = $this->getDataGenerator()->create_course(array('category' => $cat1->id,
|
||
'fullname' => 'Test 1', 'overviewfiles_filemanager' => $dratid1));
|
||
// Create course c2 with two image files (only one file will be added because of settings).
|
||
$dratid2 = $this->fill_draft_area(array('filename21.jpg' => 'Test file contents21', 'filename22.jpg' => 'Test file contents22'));
|
||
$c2 = $this->getDataGenerator()->create_course(array('category' => $cat1->id,
|
||
'fullname' => 'Test 2', 'overviewfiles_filemanager' => $dratid2));
|
||
// Create course c3 without files.
|
||
$c3 = $this->getDataGenerator()->create_course(array('category' => $cat1->id, 'fullname' => 'Test 3'));
|
||
|
||
// Change the settings to allow multiple files of any types.
|
||
$CFG->courseoverviewfileslimit = 3;
|
||
$CFG->courseoverviewfilesext = '*';
|
||
// Create course c5 with two image files.
|
||
$dratid4 = $this->fill_draft_area(array('filename41.jpg' => 'Test file contents41', 'filename42.jpg' => 'Test file contents42'));
|
||
$c4 = $this->getDataGenerator()->create_course(array('category' => $cat1->id,
|
||
'fullname' => 'Test 4', 'overviewfiles_filemanager' => $dratid4));
|
||
// Create course c6 with non-image file.
|
||
$dratid5 = $this->fill_draft_area(array('filename51.zip' => 'Test file contents51'));
|
||
$c5 = $this->getDataGenerator()->create_course(array('category' => $cat1->id,
|
||
'fullname' => 'Test 5', 'overviewfiles_filemanager' => $dratid5));
|
||
|
||
// Reset default settings.
|
||
$CFG->courseoverviewfileslimit = 1;
|
||
$CFG->courseoverviewfilesext = '.jpg,.gif,.png';
|
||
|
||
$courses = $cat1->get_courses();
|
||
$this->assertTrue($courses[$c1->id]->has_course_overviewfiles());
|
||
$this->assertTrue($courses[$c2->id]->has_course_overviewfiles());
|
||
$this->assertFalse($courses[$c3->id]->has_course_overviewfiles());
|
||
$this->assertTrue($courses[$c4->id]->has_course_overviewfiles());
|
||
$this->assertTrue($courses[$c5->id]->has_course_overviewfiles()); // Does not validate the filetypes.
|
||
|
||
$this->assertEquals(1, count($courses[$c1->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(1, count($courses[$c2->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(0, count($courses[$c3->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(1, count($courses[$c4->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(0, count($courses[$c5->id]->get_course_overviewfiles())); // Validate the filetypes.
|
||
|
||
// Overview files are not allowed, all functions return empty values.
|
||
$CFG->courseoverviewfileslimit = 0;
|
||
|
||
$this->assertFalse($courses[$c1->id]->has_course_overviewfiles());
|
||
$this->assertFalse($courses[$c2->id]->has_course_overviewfiles());
|
||
$this->assertFalse($courses[$c3->id]->has_course_overviewfiles());
|
||
$this->assertFalse($courses[$c4->id]->has_course_overviewfiles());
|
||
$this->assertFalse($courses[$c5->id]->has_course_overviewfiles());
|
||
|
||
$this->assertEquals(0, count($courses[$c1->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(0, count($courses[$c2->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(0, count($courses[$c3->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(0, count($courses[$c4->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(0, count($courses[$c5->id]->get_course_overviewfiles()));
|
||
|
||
// Multiple overview files are allowed but still limited to images.
|
||
$CFG->courseoverviewfileslimit = 3;
|
||
|
||
$this->assertTrue($courses[$c1->id]->has_course_overviewfiles());
|
||
$this->assertTrue($courses[$c2->id]->has_course_overviewfiles());
|
||
$this->assertFalse($courses[$c3->id]->has_course_overviewfiles());
|
||
$this->assertTrue($courses[$c4->id]->has_course_overviewfiles());
|
||
$this->assertTrue($courses[$c5->id]->has_course_overviewfiles()); // Still does not validate the filetypes.
|
||
|
||
$this->assertEquals(1, count($courses[$c1->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(1, count($courses[$c2->id]->get_course_overviewfiles())); // Only 1 file was actually added.
|
||
$this->assertEquals(0, count($courses[$c3->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(2, count($courses[$c4->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(0, count($courses[$c5->id]->get_course_overviewfiles()));
|
||
|
||
// Multiple overview files of any type are allowed.
|
||
$CFG->courseoverviewfilesext = '*';
|
||
|
||
$this->assertTrue($courses[$c1->id]->has_course_overviewfiles());
|
||
$this->assertTrue($courses[$c2->id]->has_course_overviewfiles());
|
||
$this->assertFalse($courses[$c3->id]->has_course_overviewfiles());
|
||
$this->assertTrue($courses[$c4->id]->has_course_overviewfiles());
|
||
$this->assertTrue($courses[$c5->id]->has_course_overviewfiles());
|
||
|
||
$this->assertEquals(1, count($courses[$c1->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(1, count($courses[$c2->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(0, count($courses[$c3->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(2, count($courses[$c4->id]->get_course_overviewfiles()));
|
||
$this->assertEquals(1, count($courses[$c5->id]->get_course_overviewfiles()));
|
||
}
|
||
|
||
/**
|
||
* Creates a draft area for current user and fills it with fake files
|
||
*
|
||
* @param array $files array of files that need to be added to filearea, filename => filecontents
|
||
* @return int draftid for the filearea
|
||
*/
|
||
protected function fill_draft_area(array $files) {
|
||
global $USER;
|
||
$usercontext = context_user::instance($USER->id);
|
||
$draftid = file_get_unused_draft_itemid();
|
||
foreach ($files as $filename => $filecontents) {
|
||
// Add actual file there.
|
||
$filerecord = array('component' => 'user', 'filearea' => 'draft',
|
||
'contextid' => $usercontext->id, 'itemid' => $draftid,
|
||
'filename' => $filename, 'filepath' => '/');
|
||
$fs = get_file_storage();
|
||
$fs->create_file_from_string($filerecord, $filecontents);
|
||
}
|
||
return $draftid;
|
||
}
|
||
}
|