Search moodle.org's
Developer Documentation

See Release Notes
Long Term Support Release

  • Bug fixes for general core bugs in 4.1.x will end 13 November 2023 (12 months).
  • Bug fixes for security issues in 4.1.x will end 10 November 2025 (36 months).
  • PHP version: minimum PHP 7.4.0 Note: minimum PHP version has increased since Moodle 4.0. PHP 8.0.x is supported too.
<?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/>.

namespace core_group;

> use core_customfield\field_controller; use core_group_external; > use core_external\external_api; use externallib_advanced_testcase; > use core_group\customfield\group_handler; > use core_group\customfield\grouping_handler;
defined('MOODLE_INTERNAL') || die(); global $CFG; require_once($CFG->dirroot . '/webservice/tests/helpers.php'); require_once($CFG->dirroot . '/group/externallib.php'); require_once($CFG->dirroot . '/group/lib.php'); /** * Group external PHPunit tests * * @package core_group * @category external * @copyright 2012 Jerome Mouneyrac * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later * @since Moodle 2.4
> * @covers \core_group_external
*/ class externallib_test extends externallib_advanced_testcase { /**
> * Create group custom field for testing. * Test create_groups > * */ > * @return field_controller public function test_create_groups() { > */ global $DB; > protected function create_group_custom_field(): field_controller { > $fieldcategory = self::getDataGenerator()->create_custom_field_category([ $this->resetAfterTest(true); > 'component' => 'core_group', > 'area' => 'group', $course = self::getDataGenerator()->create_course(); > ]); > $group1 = array(); > return self::getDataGenerator()->create_custom_field([ $group1['courseid'] = $course->id; > 'shortname' => 'testgroupcustomfield1', $group1['name'] = 'Group Test 1'; > 'type' => 'text', $group1['description'] = 'Group Test 1 description'; > 'categoryid' => $fieldcategory->get('id'), $group1['descriptionformat'] = FORMAT_MOODLE; > ]); $group1['enrolmentkey'] = 'Test group enrol secret phrase'; > } $group1['idnumber'] = 'TEST1'; > /** $group2 = array(); > * Create grouping custom field for testing. $group2['courseid'] = $course->id; > * $group2['name'] = 'Group Test 2'; > * @return field_controller $group2['description'] = 'Group Test 2 description'; > */ $group3 = array(); > protected function create_grouping_custom_field(): field_controller { $group3['courseid'] = $course->id; > $fieldcategory = self::getDataGenerator()->create_custom_field_category([ $group3['name'] = 'Group Test 3'; > 'component' => 'core_group', $group3['description'] = 'Group Test 3 description'; > 'area' => 'grouping', $group3['idnumber'] = 'TEST1'; > ]); $group4 = array(); > $group4['courseid'] = $course->id; > return self::getDataGenerator()->create_custom_field([ $group4['name'] = 'Group Test 4'; > 'shortname' => 'testgroupingcustomfield1', $group4['description'] = 'Group Test 4 description'; > 'type' => 'text', > 'categoryid' => $fieldcategory->get('id'), // Set the required capabilities by the external function > ]); $context = \context_course::instance($course->id); > } $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); > $this->assignUserCapability('moodle/course:view', $context->id, $roleid); > /**
> $group2['visibility'] = GROUPS_VISIBILITY_MEMBERS; // Call the external function. > $group2['participation'] = false;
$groups = core_group_external::create_groups(array($group1, $group2)); // We need to execute the return values cleaning process to simulate the web service server.
< $groups = \external_api::clean_returnvalue(core_group_external::create_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::create_groups_returns(), $groups);
// Checks against DB values $this->assertEquals(2, count($groups)); foreach ($groups as $group) { $dbgroup = $DB->get_record('groups', array('id' => $group['id']), '*', MUST_EXIST); switch ($dbgroup->name) { case $group1['name']: $groupdescription = $group1['description']; $groupcourseid = $group1['courseid']; $this->assertEquals($dbgroup->descriptionformat, $group1['descriptionformat']); $this->assertEquals($dbgroup->enrolmentkey, $group1['enrolmentkey']); $this->assertEquals($dbgroup->idnumber, $group1['idnumber']);
> // The visibility and participation attributes were not specified, so should match the default values. break; > $groupvisibility = GROUPS_VISIBILITY_ALL; case $group2['name']: > $groupparticipation = true;
$groupdescription = $group2['description']; $groupcourseid = $group2['courseid'];
> $groupvisibility = $group2['visibility']; break; > $groupparticipation = $group2['participation'];
default: throw new \moodle_exception('unknowgroupname'); break; } $this->assertEquals($dbgroup->description, $groupdescription); $this->assertEquals($dbgroup->courseid, $groupcourseid);
> $this->assertEquals($dbgroup->visibility, $groupvisibility); } > $this->assertEquals($dbgroup->participation, $groupparticipation);
try { $froups = core_group_external::create_groups(array($group3)); $this->fail('Exception expected due to already existing idnumber.'); } catch (\moodle_exception $e) { $this->assertInstanceOf('moodle_exception', $e); $this->assertEquals(get_string('idnumbertaken', 'error'), $e->getMessage()); } // Call without required capability $this->unassignUserCapability('moodle/course:managegroups', $context->id, $roleid); $this->expectException(\required_capability_exception::class); $froups = core_group_external::create_groups(array($group4)); } /**
> * Test create_groups with custom fields. * Test update_groups > */ */ > public function test_create_groups_with_customfields() { public function test_update_groups() { > global $DB; global $DB; > > $this->resetAfterTest(); $this->resetAfterTest(true); > $this->setAdminUser(); > $course = self::getDataGenerator()->create_course(); > $course = self::getDataGenerator()->create_course(); > $this->create_group_custom_field(); $group1data = array(); > $group = [ $group1data['courseid'] = $course->id; > 'courseid' => $course->id, $group1data['name'] = 'Group Test 1'; > 'name' => 'Create groups test (with custom fields)', $group1data['description'] = 'Group Test 1 description'; > 'description' => 'Description for create groups test with custom fields', $group1data['descriptionformat'] = FORMAT_MOODLE; > 'customfields' => [ $group1data['enrolmentkey'] = 'Test group enrol secret phrase'; > [ $group1data['idnumber'] = 'TEST1'; > 'shortname' => 'testgroupcustomfield1', $group2data = array(); > 'value' => 'Test group value 1', $group2data['courseid'] = $course->id; > ], $group2data['name'] = 'Group Test 2'; > ], $group2data['description'] = 'Group Test 2 description'; > ]; $group2data['idnumber'] = 'TEST2'; > $createdgroups = core_group_external::create_groups([$group]); > $createdgroups = external_api::clean_returnvalue(core_group_external::create_groups_returns(), $createdgroups); // Set the required capabilities by the external function. > $context = \context_course::instance($course->id); > $this->assertCount(1, $createdgroups); $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); > $createdgroup = reset($createdgroups); $this->assignUserCapability('moodle/course:view', $context->id, $roleid); > $dbgroup = $DB->get_record('groups', ['id' => $createdgroup['id']], '*', MUST_EXIST); > $this->assertEquals($group['name'], $dbgroup->name); // Create the test groups. > $this->assertEquals($group['description'], $dbgroup->description); $group1 = self::getDataGenerator()->create_group($group1data); > $group2 = self::getDataGenerator()->create_group($group2data); > $data = group_handler::create()->export_instance_data_object($createdgroup['id'], true); > $this->assertEquals('Test group value 1', $data->testgroupcustomfield1); $group1data['id'] = $group1->id; > } unset($group1data['courseid']); > $group2data['id'] = $group2->id; > /** unset($group2data['courseid']); > * Test that creating a group with an invalid visibility value throws an exception. > * // No exceptions should be triggered. > * @covers \core_group_external::create_groups $group1data['idnumber'] = 'CHANGED'; > * @return void core_group_external::update_groups(array($group1data)); > */ $group2data['description'] = 'Group Test 2 description CHANGED'; > public function test_create_group_invalid_visibility(): void { core_group_external::update_groups(array($group2data)); > $this->resetAfterTest(true); > foreach ([$group1, $group2] as $group) { > $course = self::getDataGenerator()->create_course(); $dbgroup = $DB->get_record('groups', array('id' => $group->id), '*', MUST_EXIST); > switch ($dbgroup->name) { > $group1 = array(); case $group1data['name']: > $group1['courseid'] = $course->id; $this->assertEquals($dbgroup->idnumber, $group1data['idnumber']); > $group1['name'] = 'Group Test 1'; $groupdescription = $group1data['description']; > $group1['description'] = 'Group Test 1 description'; break; > $group1['visibility'] = 1000; case $group2data['name']: > $this->assertEquals($dbgroup->idnumber, $group2data['idnumber']); > // Set the required capabilities by the external function. $groupdescription = $group2data['description']; > $context = \context_course::instance($course->id); break; > $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); default: > $this->assignUserCapability('moodle/course:view', $context->id, $roleid); throw new \moodle_exception('unknowngroupname'); > break; > // Call the external function. } > $this->expectException('invalid_parameter_exception'); $this->assertEquals($dbgroup->description, $groupdescription); > core_group_external::create_groups([$group1]); } > } > // Taken idnumber exception. > /**
$group1data['idnumber'] = 'TEST2';
> $group2data['visibility'] = GROUPS_VISIBILITY_MEMBERS;
try {
> // Visibility was not specified, so should match the default value. $groups = core_group_external::update_groups(array($group1data)); > $groupvisibility = GROUPS_VISIBILITY_ALL;
$this->fail('Exception expected due to already existing idnumber.');
> $groupvisibility = $group2data['visibility'];
} catch (\moodle_exception $e) {
> $this->assertEquals($dbgroup->visibility, $groupvisibility);
$this->assertInstanceOf('moodle_exception', $e); $this->assertEquals(get_string('idnumbertaken', 'error'), $e->getMessage()); } // Call without required capability. $group1data['idnumber'] = 'TEST1'; $this->unassignUserCapability('moodle/course:managegroups', $context->id, $roleid); $this->expectException(\required_capability_exception::class); $groups = core_group_external::update_groups(array($group1data)); } /**
> * Test update_groups with custom fields. * Test get_groups > */ */ > public function test_update_groups_with_customfields() { public function test_get_groups() { > $this->resetAfterTest(); global $DB; > $this->setAdminUser(); > $this->resetAfterTest(true); > $course = self::getDataGenerator()->create_course(); > $this->create_group_custom_field(); $course = self::getDataGenerator()->create_course(); > $group = self::getDataGenerator()->create_group(['courseid' => $course->id]); $group1data = array(); > $group1data['courseid'] = $course->id; > $data = group_handler::create()->export_instance_data_object($group->id, true); $group1data['name'] = 'Group Test 1'; > $this->assertNull($data->testgroupcustomfield1); $group1data['description'] = 'Group Test 1 description'; > $group1data['descriptionformat'] = FORMAT_MOODLE; > $updategroup = [ $group1data['enrolmentkey'] = 'Test group enrol secret phrase'; > 'id' => $group->id, $group1data['idnumber'] = 'TEST1'; > 'name' => $group->name, $group2data = array(); > 'customfields' => [ $group2data['courseid'] = $course->id; > [ $group2data['name'] = 'Group Test 2'; > 'shortname' => 'testgroupcustomfield1', $group2data['description'] = 'Group Test 2 description'; > 'value' => 'Test value 1', $group1 = self::getDataGenerator()->create_group($group1data); > ], $group2 = self::getDataGenerator()->create_group($group2data); > ], > ]; // Set the required capabilities by the external function > core_group_external::update_groups([$updategroup]); $context = \context_course::instance($course->id); > $data = group_handler::create()->export_instance_data_object($group->id, true); $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); > $this->assertEquals('Test value 1', $data->testgroupcustomfield1); $this->assignUserCapability('moodle/course:view', $context->id, $roleid); > } > // Call the external function. > /** $groups = core_group_external::get_groups(array($group1->id, $group2->id)); > * Test an exception is thrown when an invalid visibility value is passed in an update. > * // We need to execute the return values cleaning process to simulate the web service server. > * @covers \core_group_external::update_groups $groups = \external_api::clean_returnvalue(core_group_external::get_groups_returns(), $groups); > * @return void > */ // Checks against DB values > public function test_update_groups_invalid_visibility(): void { $this->assertEquals(2, count($groups)); > $this->resetAfterTest(true); foreach ($groups as $group) { > $dbgroup = $DB->get_record('groups', array('id' => $group['id']), '*', MUST_EXIST); > $course = self::getDataGenerator()->create_course(); switch ($dbgroup->name) { > case $group1->name: > $group1data = array(); $groupdescription = $group1->description; > $group1data['courseid'] = $course->id; $groupcourseid = $group1->courseid; > $group1data['name'] = 'Group Test 1'; $this->assertEquals($dbgroup->descriptionformat, $group1->descriptionformat); > $this->assertEquals($dbgroup->enrolmentkey, $group1->enrolmentkey); > // Set the required capabilities by the external function. $this->assertEquals($dbgroup->idnumber, $group1->idnumber); > $context = \context_course::instance($course->id); break; > $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); case $group2->name: > $this->assignUserCapability('moodle/course:view', $context->id, $roleid); $groupdescription = $group2->description; > $groupcourseid = $group2->courseid; > // Create the test group. break; > $group1 = self::getDataGenerator()->create_group($group1data); default: > throw new \moodle_exception('unknowgroupname'); > $group1data['id'] = $group1->id; break; > unset($group1data['courseid']); } > $group1data['visibility'] = 1000; $this->assertEquals($dbgroup->description, $groupdescription); > $this->assertEquals($dbgroup->courseid, $groupcourseid); > $this->expectException('invalid_parameter_exception'); } > core_group_external::update_groups(array($group1data)); > } // Call without required capability > $this->unassignUserCapability('moodle/course:managegroups', $context->id, $roleid); > /** > * Attempting to change the visibility of a group with members should throw an exception. $this->expectException(\required_capability_exception::class); > * $groups = core_group_external::get_groups(array($group1->id, $group2->id)); > * @covers \core_group_external::update_groups } > * @return void > */ /** > public function test_update_groups_visibility_with_members(): void { * Test delete_groups > $this->resetAfterTest(true); */ > public function test_delete_groups() { > $course = self::getDataGenerator()->create_course(); global $DB; > > $group1data = array(); $this->resetAfterTest(true); > $group1data['courseid'] = $course->id; > $group1data['name'] = 'Group Test 1'; $course = self::getDataGenerator()->create_course(); > $group1data = array(); > // Set the required capabilities by the external function. $group1data['courseid'] = $course->id; > $context = \context_course::instance($course->id); $group1data['name'] = 'Group Test 1'; > $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); $group1data['description'] = 'Group Test 1 description'; > $this->assignUserCapability('moodle/course:view', $context->id, $roleid); $group1data['descriptionformat'] = FORMAT_MOODLE; > $group1data['enrolmentkey'] = 'Test group enrol secret phrase'; > // Create the test group and add a member. $group2data = array(); > $group1 = self::getDataGenerator()->create_group($group1data); $group2data['courseid'] = $course->id; > $user1 = self::getDataGenerator()->create_and_enrol($course); $group2data['name'] = 'Group Test 2'; > self::getDataGenerator()->create_group_member(['userid' => $user1->id, 'groupid' => $group1->id]); $group2data['description'] = 'Group Test 2 description'; > $group3data['courseid'] = $course->id; > $group1data['id'] = $group1->id; $group3data['name'] = 'Group Test 3'; > unset($group1data['courseid']); $group3data['description'] = 'Group Test 3 description'; > $group1data['visibility'] = GROUPS_VISIBILITY_MEMBERS; $group1 = self::getDataGenerator()->create_group($group1data); > $group2 = self::getDataGenerator()->create_group($group2data); > $this->expectExceptionMessage('The visibility of this group cannot be changed as it currently has members.'); $group3 = self::getDataGenerator()->create_group($group3data); > core_group_external::update_groups(array($group1data)); > } // Set the required capabilities by the external function > $context = \context_course::instance($course->id); > /** $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); > * Attempting to change the participation field of a group with members should throw an exception. $this->assignUserCapability('moodle/course:view', $context->id, $roleid); > * > * @covers \core_group_external::update_groups // Checks against DB values > * @return void $groupstotal = $DB->count_records('groups', array()); > */ $this->assertEquals(3, $groupstotal); > public function test_update_groups_participation_with_members(): void { > $this->resetAfterTest(true); // Call the external function. > core_group_external::delete_groups(array($group1->id, $group2->id)); > $course = self::getDataGenerator()->create_course(); > // Checks against DB values > $group1data = array(); $groupstotal = $DB->count_records('groups', array()); > $group1data['courseid'] = $course->id; $this->assertEquals(1, $groupstotal); > $group1data['name'] = 'Group Test 1'; > // Call without required capability > // Set the required capabilities by the external function. $this->unassignUserCapability('moodle/course:managegroups', $context->id, $roleid); > $context = \context_course::instance($course->id); > $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); $this->expectException(\required_capability_exception::class); > $this->assignUserCapability('moodle/course:view', $context->id, $roleid); $froups = core_group_external::delete_groups(array($group3->id)); > } > // Create the test group and add a member. > $group1 = self::getDataGenerator()->create_group($group1data); /** > $user1 = self::getDataGenerator()->create_and_enrol($course); * Test create and update groupings. > self::getDataGenerator()->create_group_member(['userid' => $user1->id, 'groupid' => $group1->id]); * @return void > */ > $group1data['id'] = $group1->id; public function test_create_update_groupings() { > unset($group1data['courseid']); global $DB; > $group1data['participation'] = false; > $this->resetAfterTest(true); > $this->expectExceptionMessage('The participation mode of this group cannot be changed as it currently has members.'); > core_group_external::update_groups(array($group1data)); $this->setAdminUser(); > } > $course = self::getDataGenerator()->create_course(); > /**
> $group2data['visibility'] = GROUPS_VISIBILITY_MEMBERS; $grouping1data = array(); > $group2data['participation'] = false;
< $groups = \external_api::clean_returnvalue(core_group_external::get_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_groups_returns(), $groups);
$grouping1data['name'] = 'Grouping 1 Test';
> // The visibility and participation attributes were not specified, so should match the default values. $grouping1data['description'] = 'Grouping 1 Test description'; > $groupvisibility = GROUPS_VISIBILITY_ALL; $grouping1data['descriptionformat'] = FORMAT_MOODLE; > $groupparticipation = true;
$grouping1data['idnumber'] = 'TEST';
> $groupvisibility = $group2->visibility; > $groupparticipation = $group2->participation;
$grouping1 = self::getDataGenerator()->create_grouping($grouping1data);
> $this->assertEquals($dbgroup->visibility, $groupvisibility); > $this->assertEquals($dbgroup->participation, $groupparticipation);
$grouping1data['name'] = 'Another group';
> * Test get_groups with customfields. > */ try { > public function test_get_groups_with_customfields() { $groupings = core_group_external::create_groupings(array($grouping1data)); > $this->resetAfterTest(); $this->fail('Exception expected due to already existing idnumber.'); > $this->setAdminUser(); } catch (\moodle_exception $e) { > $this->assertInstanceOf('moodle_exception', $e); > $course = self::getDataGenerator()->create_course(); $this->assertEquals(get_string('idnumbertaken', 'error'), $e->getMessage()); > $this->create_group_custom_field(); } > $group = self::getDataGenerator()->create_group([ > 'courseid' => $course->id, // No exception should be triggered. > 'customfield_testgroupcustomfield1' => 'Test group value 1', $grouping1data['id'] = $grouping1->id; > ]); $grouping1data['idnumber'] = 'CHANGED'; > unset($grouping1data['courseid']); > // Call the external function. core_group_external::update_groupings(array($grouping1data)); > $groups = core_group_external::get_groups([$group->id]); > // We need to execute the return values cleaning process to simulate the web service server. $grouping2data = array(); > $groups = external_api::clean_returnvalue(core_group_external::get_groups_returns(), $groups); $grouping2data['courseid'] = $course->id; > $grouping2data['name'] = 'Grouping 2 Test'; > $this->assertEquals(1, count($groups)); $grouping2data['description'] = 'Grouping 2 Test description'; > $groupresult = reset($groups); $grouping2data['descriptionformat'] = FORMAT_MOODLE; > $this->assertEquals(1, count($groupresult['customfields'])); $grouping2data['idnumber'] = 'TEST'; > $customfield = reset($groupresult['customfields']); > $this->assertEquals('testgroupcustomfield1', $customfield['shortname']); $grouping2 = self::getDataGenerator()->create_grouping($grouping2data); > $this->assertEquals('Test group value 1', $customfield['value']); > } $grouping2data['id'] = $grouping2->id; > $grouping2data['idnumber'] = 'CHANGED'; > /**
unset($grouping2data['courseid']); try { $groupings = core_group_external::update_groupings(array($grouping2data)); $this->fail('Exception expected due to already existing idnumber.'); } catch (\moodle_exception $e) { $this->assertInstanceOf('moodle_exception', $e); $this->assertEquals(get_string('idnumbertaken', 'error'), $e->getMessage()); } } /**
> * Test create_groupings with custom fields. * Test get_groupings > */ */ > public function test_create_groupings_with_customfields() { public function test_get_groupings() { > global $DB; global $DB; > > $this->resetAfterTest(); $this->resetAfterTest(true); > $this->setAdminUser(); > $course = self::getDataGenerator()->create_course(); > $course = self::getDataGenerator()->create_course(); > $this->create_grouping_custom_field(); $groupingdata = array(); > $grouping = [ $groupingdata['courseid'] = $course->id; > 'courseid' => $course->id, $groupingdata['name'] = 'Grouping Test'; > 'name' => 'Create groupings test (with custom fields)', $groupingdata['description'] = 'Grouping Test description'; > 'description' => 'Description for create groupings test with custom fields', $groupingdata['descriptionformat'] = FORMAT_MOODLE; > 'idnumber' => 'groupingidnumber1', > 'customfields' => [ $grouping = self::getDataGenerator()->create_grouping($groupingdata); > [ > 'shortname' => 'testgroupingcustomfield1', // Set the required capabilities by the external function. > 'value' => 'Test grouping value 1', $context = \context_course::instance($course->id); > ], $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); > ], $this->assignUserCapability('moodle/course:view', $context->id, $roleid); > ]; > $createdgroupings = core_group_external::create_groupings([$grouping]); // Call the external function without specifying the optional parameter. > $createdgroupings = external_api::clean_returnvalue(core_group_external::create_groupings_returns(), $createdgroupings); $groupings = core_group_external::get_groupings(array($grouping->id)); > // We need to execute the return values cleaning process to simulate the web service server. > $this->assertCount(1, $createdgroupings); $groupings = \external_api::clean_returnvalue(core_group_external::get_groupings_returns(), $groupings); > $createdgrouping = reset($createdgroupings); > $dbgroup = $DB->get_record('groupings', ['id' => $createdgrouping['id']], '*', MUST_EXIST); $this->assertEquals(1, count($groupings)); > $this->assertEquals($grouping['name'], $dbgroup->name); > $this->assertEquals($grouping['description'], $dbgroup->description); $group1data = array(); > $this->assertEquals($grouping['idnumber'], $dbgroup->idnumber); $group1data['courseid'] = $course->id; > $group1data['name'] = 'Group Test 1'; > $data = grouping_handler::create()->export_instance_data_object($createdgrouping['id'], true); $group1data['description'] = 'Group Test 1 description'; > $this->assertEquals('Test grouping value 1', $data->testgroupingcustomfield1); $group1data['descriptionformat'] = FORMAT_MOODLE; > } $group2data = array(); > $group2data['courseid'] = $course->id; > /** $group2data['name'] = 'Group Test 2'; > * Test update_groups with custom fields. $group2data['description'] = 'Group Test 2 description'; > */ $group2data['descriptionformat'] = FORMAT_MOODLE; > public function test_update_groupings_with_customfields() { > $this->resetAfterTest(); $group1 = self::getDataGenerator()->create_group($group1data); > $this->setAdminUser(); $group2 = self::getDataGenerator()->create_group($group2data); > > $course = self::getDataGenerator()->create_course(); groups_assign_grouping($grouping->id, $group1->id); > $this->create_grouping_custom_field(); groups_assign_grouping($grouping->id, $group2->id); > $grouping = self::getDataGenerator()->create_grouping(['courseid' => $course->id]); > // Call the external function specifying that groups are returned. > $data = grouping_handler::create()->export_instance_data_object($grouping->id, true); $groupings = core_group_external::get_groupings(array($grouping->id), true); > $this->assertNull($data->testgroupingcustomfield1); // We need to execute the return values cleaning process to simulate the web service server. > $groupings = \external_api::clean_returnvalue(core_group_external::get_groupings_returns(), $groupings); > $updategroup = [ $this->assertEquals(1, count($groupings)); > 'id' => $grouping->id, $this->assertEquals(2, count($groupings[0]['groups'])); > 'name' => $grouping->name, foreach ($groupings[0]['groups'] as $group) { > 'description' => $grouping->description, $dbgroup = $DB->get_record('groups', array('id' => $group['id']), '*', MUST_EXIST); > 'customfields' => [ $dbgroupinggroups = $DB->get_record('groupings_groups', > [ array('groupingid' => $groupings[0]['id'], > 'shortname' => 'testgroupingcustomfield1', 'groupid' => $group['id']), > 'value' => 'Test grouping value 1', '*', MUST_EXIST); > ], switch ($dbgroup->name) { > ], case $group1->name: > ]; $groupdescription = $group1->description; > core_group_external::update_groupings([$updategroup]); $groupcourseid = $group1->courseid; > $data = grouping_handler::create()->export_instance_data_object($grouping->id, true); break; > $this->assertEquals('Test grouping value 1', $data->testgroupingcustomfield1); case $group2->name: > } $groupdescription = $group2->description; > $groupcourseid = $group2->courseid; > /**
< $groupings = \external_api::clean_returnvalue(core_group_external::get_groupings_returns(), $groupings);
> $groupings = external_api::clean_returnvalue(core_group_external::get_groupings_returns(), $groupings);
< $groupings = \external_api::clean_returnvalue(core_group_external::get_groupings_returns(), $groupings);
> $groupings = external_api::clean_returnvalue(core_group_external::get_groupings_returns(), $groupings);
throw new \moodle_exception('unknowgroupname'); break; } $this->assertEquals($dbgroup->description, $groupdescription); $this->assertEquals($dbgroup->courseid, $groupcourseid); } } /**
> * Test get_groupings with customfields. * Test delete_groupings. > */ */ > public function test_get_groupings_with_customfields() { public function test_delete_groupings() { > $this->resetAfterTest(); global $DB; > $this->setAdminUser(); > $this->resetAfterTest(true); > $course = self::getDataGenerator()->create_course(); > $this->create_grouping_custom_field(); $course = self::getDataGenerator()->create_course(); > $grouping = self::getDataGenerator()->create_grouping([ > 'courseid' => $course->id, $groupingdata1 = array(); > 'customfield_testgroupingcustomfield1' => 'Test grouping value 1', $groupingdata1['courseid'] = $course->id; > ]); $groupingdata1['name'] = 'Grouping Test'; > $this->create_group_custom_field(); $groupingdata1['description'] = 'Grouping Test description'; > $group = self::getDataGenerator()->create_group([ $groupingdata1['descriptionformat'] = FORMAT_MOODLE; > 'courseid' => $course->id, $groupingdata2 = array(); > 'customfield_testgroupcustomfield1' => 'Test group value 1', $groupingdata2['courseid'] = $course->id; > ]); $groupingdata2['name'] = 'Grouping Test'; > groups_assign_grouping($grouping->id, $group->id); $groupingdata2['description'] = 'Grouping Test description'; > $groupingdata2['descriptionformat'] = FORMAT_MOODLE; > // Call the external function. $groupingdata3 = array(); > $groupings = core_group_external::get_groupings([$grouping->id]); $groupingdata3['courseid'] = $course->id; > // We need to execute the return values cleaning process to simulate the web service server. $groupingdata3['name'] = 'Grouping Test'; > $groupings = external_api::clean_returnvalue(core_group_external::get_groupings_returns(), $groupings); $groupingdata3['description'] = 'Grouping Test description'; > $groupingdata3['descriptionformat'] = FORMAT_MOODLE; > $this->assertEquals(1, count($groupings)); > $groupingresult = reset($groupings); $grouping1 = self::getDataGenerator()->create_grouping($groupingdata1); > $this->assertEquals(1, count($groupingresult['customfields'])); $grouping2 = self::getDataGenerator()->create_grouping($groupingdata2); > $customfield = reset($groupingresult['customfields']); $grouping3 = self::getDataGenerator()->create_grouping($groupingdata3); > $this->assertEquals('testgroupingcustomfield1', $customfield['shortname']); > $this->assertEquals('Test grouping value 1', $customfield['value']); // Set the required capabilities by the external function. > $this->assertArrayNotHasKey('groups', $groupingresult); $context = \context_course::instance($course->id); > $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); > // Call the external function with return group parameter. $this->assignUserCapability('moodle/course:view', $context->id, $roleid); > $groupings = core_group_external::get_groupings([$grouping->id], true); > // We need to execute the return values cleaning process to simulate the web service server. // Checks against DB values. > $groupings = external_api::clean_returnvalue(core_group_external::get_groupings_returns(), $groupings); $groupingstotal = $DB->count_records('groupings', array()); > $this->assertEquals(3, $groupingstotal); > $this->assertEquals(1, count($groupings)); > $groupingresult = reset($groupings); // Call the external function. > $this->assertEquals(1, count($groupingresult['customfields'])); core_group_external::delete_groupings(array($grouping1->id, $grouping2->id)); > $this->assertArrayHasKey('groups', $groupingresult); > $this->assertEquals(1, count($groupingresult['groups'])); // Checks against DB values. > $groupresult = reset($groupingresult['groups']); $groupingstotal = $DB->count_records('groupings', array()); > $this->assertEquals(1, count($groupresult['customfields'])); $this->assertEquals(1, $groupingstotal); > $customfield = reset($groupresult['customfields']); > $this->assertEquals('testgroupcustomfield1', $customfield['shortname']); // Call without required capability. > $this->assertEquals('Test group value 1', $customfield['value']); $this->unassignUserCapability('moodle/course:managegroups', $context->id, $roleid); > } > $this->expectException(\required_capability_exception::class); > /**
core_group_external::delete_groupings(array($grouping3->id)); } /** * Test get_groups */ public function test_get_course_user_groups() { global $DB; $this->resetAfterTest(true); $student1 = self::getDataGenerator()->create_user(); $student2 = self::getDataGenerator()->create_user(); $teacher = self::getDataGenerator()->create_user(); $course = self::getDataGenerator()->create_course(); $anothercourse = self::getDataGenerator()->create_course(); $emptycourse = self::getDataGenerator()->create_course(); $studentrole = $DB->get_record('role', array('shortname' => 'student')); $this->getDataGenerator()->enrol_user($student1->id, $course->id, $studentrole->id); $this->getDataGenerator()->enrol_user($student1->id, $anothercourse->id, $studentrole->id); $this->getDataGenerator()->enrol_user($student2->id, $course->id, $studentrole->id); $teacherrole = $DB->get_record('role', array('shortname' => 'editingteacher')); $this->getDataGenerator()->enrol_user($teacher->id, $course->id, $teacherrole->id); $this->getDataGenerator()->enrol_user($teacher->id, $emptycourse->id, $teacherrole->id); $group1data = array(); $group1data['courseid'] = $course->id; $group1data['name'] = 'Group Test 1'; $group1data['description'] = 'Group Test 1 description'; $group1data['idnumber'] = 'TEST1'; $group2data = array(); $group2data['courseid'] = $course->id; $group2data['name'] = 'Group Test 2'; $group2data['description'] = 'Group Test 2 description'; $group3data = array(); $group3data['courseid'] = $anothercourse->id; $group3data['name'] = 'Group Test 3'; $group3data['description'] = 'Group Test 3 description'; $group3data['idnumber'] = 'TEST3'; $group1 = self::getDataGenerator()->create_group($group1data); $group2 = self::getDataGenerator()->create_group($group2data); $group3 = self::getDataGenerator()->create_group($group3data); groups_add_member($group1->id, $student1->id); groups_add_member($group1->id, $student2->id); groups_add_member($group2->id, $student1->id); groups_add_member($group3->id, $student1->id); // Create a grouping. $groupingdata = array(); $groupingdata['courseid'] = $course->id; $groupingdata['name'] = 'Grouping Test'; $groupingdata['description'] = 'Grouping Test description'; $groupingdata['descriptionformat'] = FORMAT_MOODLE; $grouping = self::getDataGenerator()->create_grouping($groupingdata); // Grouping only containing group1. groups_assign_grouping($grouping->id, $group1->id); $this->setUser($student1); $groups = core_group_external::get_course_user_groups($course->id, $student1->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
// Check that I see my groups. $this->assertCount(2, $groups['groups']); $this->assertEquals($course->id, $groups['groups'][0]['courseid']); $this->assertEquals($course->id, $groups['groups'][1]['courseid']); // Check that I only see my groups inside the given grouping. $groups = core_group_external::get_course_user_groups($course->id, $student1->id, $grouping->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
// Check that I see my groups in the grouping. $this->assertCount(1, $groups['groups']); $this->assertEquals($group1->id, $groups['groups'][0]['id']); // Check optional parameters (all student 1 courses and current user). $groups = core_group_external::get_course_user_groups();
< $groups = \external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
// Check that I see my groups in all my courses. $this->assertCount(3, $groups['groups']); $this->setUser($student2); $groups = core_group_external::get_course_user_groups($course->id, $student2->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
// Check that I see my groups. $this->assertCount(1, $groups['groups']); $this->assertEquals($group1data['name'], $groups['groups'][0]['name']); $this->assertEquals($group1data['description'], $groups['groups'][0]['description']); $this->assertEquals($group1data['idnumber'], $groups['groups'][0]['idnumber']); $this->setUser($teacher); $groups = core_group_external::get_course_user_groups($course->id, $student1->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
// Check that a teacher can see student groups in given course. $this->assertCount(2, $groups['groups']); $groups = core_group_external::get_course_user_groups($course->id, $student2->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
// Check that a teacher can see student groups in given course. $this->assertCount(1, $groups['groups']); $groups = core_group_external::get_course_user_groups(0, $student1->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
// Check that a teacher can see student groups in all the user courses if the teacher is enrolled in the course. $this->assertCount(2, $groups['groups']); // Teacher only see groups in first course. $this->assertCount(1, $groups['warnings']); // Enrolment warnings. $this->assertEquals('1', $groups['warnings'][0]['warningcode']); // Enrol teacher in second course. $this->getDataGenerator()->enrol_user($teacher->id, $anothercourse->id, $teacherrole->id); $groups = core_group_external::get_course_user_groups(0, $student1->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
// Check that a teacher can see student groups in all the user courses if the teacher is enrolled in the course. $this->assertCount(3, $groups['groups']); // Check permissions. $this->setUser($student1); // Student can's see other students group. $groups = core_group_external::get_course_user_groups($course->id, $student2->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
$this->assertCount(1, $groups['warnings']); $this->assertEquals('cannotmanagegroups', $groups['warnings'][0]['warningcode']); // Not enrolled course. $groups = core_group_external::get_course_user_groups($emptycourse->id, $student2->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
$this->assertCount(1, $groups['warnings']); $this->assertEquals('1', $groups['warnings'][0]['warningcode']); $this->setUser($teacher); // Check user checking not enrolled in given course. $groups = core_group_external::get_course_user_groups($emptycourse->id, $student1->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_course_user_groups_returns(), $groups);
$this->assertCount(1, $groups['warnings']); $this->assertEquals('notenrolled', $groups['warnings'][0]['warningcode']); } /** * Test get_activity_allowed_groups */ public function test_get_activity_allowed_groups() { global $DB; $this->resetAfterTest(true); $generator = self::getDataGenerator(); $student = $generator->create_user(); $otherstudent = $generator->create_user(); $teacher = $generator->create_user(); $course = $generator->create_course(); $othercourse = $generator->create_course(); $studentrole = $DB->get_record('role', array('shortname' => 'student')); $teacherrole = $DB->get_record('role', array('shortname' => 'editingteacher')); $generator->enrol_user($student->id, $course->id, $studentrole->id); $generator->enrol_user($otherstudent->id, $othercourse->id, $studentrole->id); $generator->enrol_user($teacher->id, $course->id, $teacherrole->id); $forum1 = $generator->create_module("forum", array('course' => $course->id), array('groupmode' => VISIBLEGROUPS)); $forum2 = $generator->create_module("forum", array('course' => $othercourse->id)); $forum3 = $generator->create_module("forum", array('course' => $course->id), array('visible' => 0)); // Request data for tests. $cm1 = get_coursemodule_from_instance("forum", $forum1->id); $cm2 = get_coursemodule_from_instance("forum", $forum2->id); $cm3 = get_coursemodule_from_instance("forum", $forum3->id); $group1data = array(); $group1data['courseid'] = $course->id; $group1data['name'] = 'Group Test 1'; $group1data['description'] = 'Group Test 1 description'; $group1data['idnumber'] = 'TEST1'; $group2data = array(); $group2data['courseid'] = $course->id; $group2data['name'] = 'Group Test 2'; $group2data['description'] = 'Group Test 2 description'; $group2data['idnumber'] = 'TEST2'; $group1 = $generator->create_group($group1data); $group2 = $generator->create_group($group2data); groups_add_member($group1->id, $student->id); groups_add_member($group2->id, $student->id); $this->setUser($student); // First try possible errors. try { $data = core_group_external::get_activity_allowed_groups($cm2->id); } catch (\moodle_exception $e) { $this->assertEquals('requireloginerror', $e->errorcode); } try { $data = core_group_external::get_activity_allowed_groups($cm3->id); } catch (\moodle_exception $e) { $this->assertEquals('requireloginerror', $e->errorcode); } // Retrieve my groups. $groups = core_group_external::get_activity_allowed_groups($cm1->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_activity_allowed_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_activity_allowed_groups_returns(), $groups);
$this->assertCount(2, $groups['groups']); $this->assertFalse($groups['canaccessallgroups']); foreach ($groups['groups'] as $group) { if ($group['name'] == $group1data['name']) { $this->assertEquals($group1data['description'], $group['description']); $this->assertEquals($group1data['idnumber'], $group['idnumber']); } else { $this->assertEquals($group2data['description'], $group['description']); $this->assertEquals($group2data['idnumber'], $group['idnumber']); } } $this->setUser($teacher); // Retrieve other users groups. $groups = core_group_external::get_activity_allowed_groups($cm1->id, $student->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_activity_allowed_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_activity_allowed_groups_returns(), $groups);
$this->assertCount(2, $groups['groups']); // We are checking the $student passed as parameter so this will return false. $this->assertFalse($groups['canaccessallgroups']); // Check warnings. Trying to get groups for a user not enrolled in course. $groups = core_group_external::get_activity_allowed_groups($cm1->id, $otherstudent->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_activity_allowed_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_activity_allowed_groups_returns(), $groups);
$this->assertCount(1, $groups['warnings']); $this->assertFalse($groups['canaccessallgroups']); // Checking teacher groups. $groups = core_group_external::get_activity_allowed_groups($cm1->id);
< $groups = \external_api::clean_returnvalue(core_group_external::get_activity_allowed_groups_returns(), $groups);
> $groups = external_api::clean_returnvalue(core_group_external::get_activity_allowed_groups_returns(), $groups);
$this->assertCount(2, $groups['groups']); // Teachers by default can access all groups. $this->assertTrue($groups['canaccessallgroups']); } /** * Test get_activity_groupmode */ public function test_get_activity_groupmode() { global $DB; $this->resetAfterTest(true); $generator = self::getDataGenerator(); $student = $generator->create_user(); $course = $generator->create_course(); $othercourse = $generator->create_course(); $studentrole = $DB->get_record('role', array('shortname' => 'student')); $generator->enrol_user($student->id, $course->id, $studentrole->id); $forum1 = $generator->create_module("forum", array('course' => $course->id), array('groupmode' => VISIBLEGROUPS)); $forum2 = $generator->create_module("forum", array('course' => $othercourse->id)); $forum3 = $generator->create_module("forum", array('course' => $course->id), array('visible' => 0)); // Request data for tests. $cm1 = get_coursemodule_from_instance("forum", $forum1->id); $cm2 = get_coursemodule_from_instance("forum", $forum2->id); $cm3 = get_coursemodule_from_instance("forum", $forum3->id); $this->setUser($student); $data = core_group_external::get_activity_groupmode($cm1->id);
< $data = \external_api::clean_returnvalue(core_group_external::get_activity_groupmode_returns(), $data);
> $data = external_api::clean_returnvalue(core_group_external::get_activity_groupmode_returns(), $data);
$this->assertEquals(VISIBLEGROUPS, $data['groupmode']); try { $data = core_group_external::get_activity_groupmode($cm2->id); } catch (\moodle_exception $e) { $this->assertEquals('requireloginerror', $e->errorcode); } try { $data = core_group_external::get_activity_groupmode($cm3->id); } catch (\moodle_exception $e) { $this->assertEquals('requireloginerror', $e->errorcode); } } /** * Test add_group_members. */ public function test_add_group_members() { global $DB; $this->resetAfterTest(true); $student1 = self::getDataGenerator()->create_user(); $student2 = self::getDataGenerator()->create_user(); $student3 = self::getDataGenerator()->create_user(); $course = self::getDataGenerator()->create_course(); $studentrole = $DB->get_record('role', array('shortname' => 'student')); $this->getDataGenerator()->enrol_user($student1->id, $course->id, $studentrole->id); $this->getDataGenerator()->enrol_user($student2->id, $course->id, $studentrole->id); $this->getDataGenerator()->enrol_user($student3->id, $course->id, $studentrole->id); $group1data = array(); $group1data['courseid'] = $course->id; $group1data['name'] = 'Group Test 1'; $group1data['description'] = 'Group Test 1 description'; $group1data['idnumber'] = 'TEST1'; $group1 = self::getDataGenerator()->create_group($group1data); // Checks against DB values. $memberstotal = $DB->count_records('groups_members', ['groupid' => $group1->id]); $this->assertEquals(0, $memberstotal); // Set the required capabilities by the external function. $context = \context_course::instance($course->id); $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); $this->assignUserCapability('moodle/course:view', $context->id, $roleid); core_group_external::add_group_members([ 'members' => [ 'groupid' => $group1->id, 'userid' => $student1->id, ] ]); core_group_external::add_group_members([ 'members' => [ 'groupid' => $group1->id, 'userid' => $student2->id, ] ]); core_group_external::add_group_members([ 'members' => [ 'groupid' => $group1->id, 'userid' => $student3->id, ] ]); // Checks against DB values. $memberstotal = $DB->count_records('groups_members', ['groupid' => $group1->id]); $this->assertEquals(3, $memberstotal); } /** * Test delete_group_members. */ public function test_delete_group_members() { global $DB; $this->resetAfterTest(true); $student1 = self::getDataGenerator()->create_user(); $student2 = self::getDataGenerator()->create_user(); $student3 = self::getDataGenerator()->create_user(); $course = self::getDataGenerator()->create_course(); $studentrole = $DB->get_record('role', array('shortname' => 'student')); $this->getDataGenerator()->enrol_user($student1->id, $course->id, $studentrole->id); $this->getDataGenerator()->enrol_user($student2->id, $course->id, $studentrole->id); $this->getDataGenerator()->enrol_user($student3->id, $course->id, $studentrole->id); $group1data = array(); $group1data['courseid'] = $course->id; $group1data['name'] = 'Group Test 1'; $group1data['description'] = 'Group Test 1 description'; $group1data['idnumber'] = 'TEST1'; $group1 = self::getDataGenerator()->create_group($group1data); groups_add_member($group1->id, $student1->id); groups_add_member($group1->id, $student2->id); groups_add_member($group1->id, $student3->id); // Checks against DB values. $memberstotal = $DB->count_records('groups_members', ['groupid' => $group1->id]); $this->assertEquals(3, $memberstotal); // Set the required capabilities by the external function. $context = \context_course::instance($course->id); $roleid = $this->assignUserCapability('moodle/course:managegroups', $context->id); $this->assignUserCapability('moodle/course:view', $context->id, $roleid); core_group_external::delete_group_members([ 'members' => [ 'groupid' => $group1->id, 'userid' => $student2->id, ] ]); // Checks against DB values. $memberstotal = $DB->count_records('groups_members', ['groupid' => $group1->id]); $this->assertEquals(2, $memberstotal); } }