<?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_cohort;
use core_cohort_external;
> use core_external\external_api;
use externallib_advanced_testcase;
> use core_cohort\customfield\cohort_handler;
defined('MOODLE_INTERNAL') || die();
global $CFG;
require_once($CFG->dirroot . '/webservice/tests/helpers.php');
require_once($CFG->dirroot . '/cohort/externallib.php');
/**
* External cohort API
*
* @package core_cohort
* @category external
* @copyright MediaTouch 2000 srl
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
class externallib_test extends externallib_advanced_testcase {
/**
> * Create cohort custom fields for testing.
* Test create_cohorts
> */
*/
> protected function create_custom_fields(): void {
public function test_create_cohorts() {
> $fieldcategory = self::getDataGenerator()->create_custom_field_category([
global $USER, $CFG, $DB;
> 'component' => 'core_cohort',
> 'area' => 'cohort',
$this->resetAfterTest(true);
> 'name' => 'Other fields',
> ]);
set_config('allowcohortthemes', 1);
> self::getDataGenerator()->create_custom_field([
> 'shortname' => 'testfield1',
$contextid = \context_system::instance()->id;
> 'name' => 'Custom field',
$category = $this->getDataGenerator()->create_category();
> 'type' => 'text',
> 'categoryid' => $fieldcategory->get('id'),
$cohort1 = array(
> ]);
'categorytype' => array('type' => 'id', 'value' => $category->id),
> self::getDataGenerator()->create_custom_field([
'name' => 'cohort test 1',
> 'shortname' => 'testfield2',
'idnumber' => 'cohorttest1',
> 'name' => 'Custom field',
'description' => 'This is a description for cohorttest1',
> 'type' => 'text',
'theme' => 'classic'
> 'categoryid' => $fieldcategory->get('id'),
);
> ]);
> }
$cohort2 = array(
>
'categorytype' => array('type' => 'system', 'value' => ''),
> /**
< global $USER, $CFG, $DB;
> global $DB;
'idnumber' => 'cohorttest2',
> // Custom fields.
'description' => 'This is a description for cohorttest2',
> $this->create_custom_fields();
'visible' => 0
>
);
$cohort3 = array(
'categorytype' => array('type' => 'id', 'value' => $category->id),
'name' => 'cohort test 3',
'idnumber' => 'cohorttest3',
'description' => 'This is a description for cohorttest3'
);
$roleid = $this->assignUserCapability('moodle/cohort:manage', $contextid);
$cohort4 = array(
'categorytype' => array('type' => 'id', 'value' => $category->id),
'name' => 'cohort test 4',
'idnumber' => 'cohorttest4',
'description' => 'This is a description for cohorttest4',
'theme' => 'classic'
);
> $cohort5 = array(
// Call the external function.
> 'categorytype' => array('type' => 'id', 'value' => $category->id),
$this->setCurrentTimeStart();
> 'name' => 'cohort test 5 (with custom fields)',
$createdcohorts = core_cohort_external::create_cohorts(array($cohort1, $cohort2));
> 'idnumber' => 'cohorttest5',
$createdcohorts = \external_api::clean_returnvalue(core_cohort_external::create_cohorts_returns(), $createdcohorts);
> 'description' => 'This is a description for cohorttest5',
> 'customfields' => array(
// Check we retrieve the good total number of created cohorts + no error on capability.
> array(
$this->assertEquals(2, count($createdcohorts));
> 'shortname' => 'testfield1',
> 'value' => 'Test value 1',
foreach ($createdcohorts as $createdcohort) {
> ),
$dbcohort = $DB->get_record('cohort', array('id' => $createdcohort['id']));
> array(
if ($createdcohort['idnumber'] == $cohort1['idnumber']) {
> 'shortname' => 'testfield2',
$conid = $DB->get_field('context', 'id', array('instanceid' => $cohort1['categorytype']['value'],
> 'value' => 'Test value 2',
'contextlevel' => CONTEXT_COURSECAT));
> ),
$this->assertEquals($dbcohort->contextid, $conid);
> ),
$this->assertEquals($dbcohort->name, $cohort1['name']);
> );
$this->assertEquals($dbcohort->description, $cohort1['description']);
>
< $createdcohorts = \external_api::clean_returnvalue(core_cohort_external::create_cohorts_returns(), $createdcohorts);
> $createdcohorts = external_api::clean_returnvalue(core_cohort_external::create_cohorts_returns(), $createdcohorts);
// As $CFG->allowcohortthemes is enabled, theme must be initialised.
$this->assertEquals($dbcohort->theme, $cohort1['theme']);
} else if ($createdcohort['idnumber'] == $cohort2['idnumber']) {
$this->assertEquals($dbcohort->contextid, \context_system::instance()->id);
$this->assertEquals($dbcohort->name, $cohort2['name']);
$this->assertEquals($dbcohort->description, $cohort2['description']);
$this->assertEquals($dbcohort->visible, $cohort2['visible']);
// Although $CFG->allowcohortthemes is enabled, no theme is defined for this cohort.
$this->assertEquals($dbcohort->theme, '');
} else {
$this->fail('Unrecognised cohort found');
}
$this->assertTimeCurrent($dbcohort->timecreated);
$this->assertTimeCurrent($dbcohort->timemodified);
}
> $createdcohorts = core_cohort_external::create_cohorts(array($cohort5));
// Call when $CFG->allowcohortthemes is disabled.
> $createdcohorts = external_api::clean_returnvalue(core_cohort_external::create_cohorts_returns(), $createdcohorts);
set_config('allowcohortthemes', 0);
>
$createdcohorts = core_cohort_external::create_cohorts(array($cohort4));
> $this->assertCount(1, $createdcohorts);
$createdcohorts = \external_api::clean_returnvalue(core_cohort_external::create_cohorts_returns(), $createdcohorts);
> $createdcohort = reset($createdcohorts);
foreach ($createdcohorts as $createdcohort) {
> $dbcohort = $DB->get_record('cohort', array('id' => $createdcohort['id']));
$dbcohort = $DB->get_record('cohort', array('id' => $createdcohort['id']));
> $this->assertEquals($cohort5['name'], $dbcohort->name);
if ($createdcohort['idnumber'] == $cohort4['idnumber']) {
> $this->assertEquals($cohort5['description'], $dbcohort->description);
$conid = $DB->get_field('context', 'id', array('instanceid' => $cohort4['categorytype']['value'],
> $this->assertEquals(1, $dbcohort->visible);
'contextlevel' => CONTEXT_COURSECAT));
> $this->assertEquals('', $dbcohort->theme);
$this->assertEquals($dbcohort->contextid, $conid);
>
$this->assertEquals($dbcohort->name, $cohort4['name']);
> $data = cohort_handler::create()->export_instance_data_object($createdcohort['id'], true);
$this->assertEquals($dbcohort->description, $cohort4['description']);
> $this->assertEquals('Test value 1', $data->testfield1);
$this->assertEquals($dbcohort->visible, 1); // Field was not specified, ensure it is visible by default.
> $this->assertEquals('Test value 2', $data->testfield2);
$this->assertEquals($dbcohort->theme, ''); // As $CFG->allowcohortthemes is disabled, theme must be empty.
>
< $createdcohorts = \external_api::clean_returnvalue(core_cohort_external::create_cohorts_returns(), $createdcohorts);
> $createdcohorts = external_api::clean_returnvalue(core_cohort_external::create_cohorts_returns(), $createdcohorts);
}
// Call without required capability.
$this->unassignUserCapability('moodle/cohort:manage', $contextid, $roleid);
$this->expectException(\required_capability_exception::class);
$createdcohorts = core_cohort_external::create_cohorts(array($cohort3));
}
/**
* Test delete_cohorts
*/
public function test_delete_cohorts() {
global $USER, $CFG, $DB;
$this->resetAfterTest(true);
$cohort1 = self::getDataGenerator()->create_cohort();
$cohort2 = self::getDataGenerator()->create_cohort();
// Check the cohorts were correctly created.
$this->assertEquals(2, $DB->count_records_select('cohort', ' (id = :cohortid1 OR id = :cohortid2)',
array('cohortid1' => $cohort1->id, 'cohortid2' => $cohort2->id)));
$contextid = $cohort1->contextid;
$roleid = $this->assignUserCapability('moodle/cohort:manage', $contextid);
// Call the external function.
core_cohort_external::delete_cohorts(array($cohort1->id, $cohort2->id));
// Check we retrieve no cohorts + no error on capability.
$this->assertEquals(0, $DB->count_records_select('cohort', ' (id = :cohortid1 OR id = :cohortid2)',
array('cohortid1' => $cohort1->id, 'cohortid2' => $cohort2->id)));
// Call without required capability.
$cohort1 = self::getDataGenerator()->create_cohort();
$cohort2 = self::getDataGenerator()->create_cohort();
$this->unassignUserCapability('moodle/cohort:manage', $contextid, $roleid);
$this->expectException(\required_capability_exception::class);
core_cohort_external::delete_cohorts(array($cohort1->id, $cohort2->id));
}
/**
* Test get_cohorts
*/
public function test_get_cohorts() {
< global $USER, $CFG;
<
$this->resetAfterTest(true);
> // Custom fields.
set_config('allowcohortthemes', 1);
> $this->create_custom_fields();
>
$cohort1 = array(
'contextid' => 1,
'name' => 'cohortnametest1',
'idnumber' => 'idnumbertest1',
'description' => 'This is a description for cohort 1',
< 'theme' => 'classic'
> 'theme' => 'classic',
> 'customfield_testfield1' => 'Test value 1',
> 'customfield_testfield2' => 'Test value 2',
);
>
$cohort1 = self::getDataGenerator()->create_cohort($cohort1);
> // We need a site admin to be able to populate cohorts custom fields.
$cohort2 = self::getDataGenerator()->create_cohort();
> $this->setAdminUser();
>
$context = \context_system::instance();
$roleid = $this->assignUserCapability('moodle/cohort:view', $context->id);
// Call the external function.
$returnedcohorts = core_cohort_external::get_cohorts(array(
$cohort1->id, $cohort2->id));
< $returnedcohorts = \external_api::clean_returnvalue(core_cohort_external::get_cohorts_returns(), $returnedcohorts);
> $returnedcohorts = external_api::clean_returnvalue(core_cohort_external::get_cohorts_returns(), $returnedcohorts);
// Check we retrieve the good total number of enrolled cohorts + no error on capability.
$this->assertEquals(2, count($returnedcohorts));
foreach ($returnedcohorts as $enrolledcohort) {
if ($enrolledcohort['idnumber'] == $cohort1->idnumber) {
$this->assertEquals($cohort1->name, $enrolledcohort['name']);
$this->assertEquals($cohort1->description, $enrolledcohort['description']);
$this->assertEquals($cohort1->visible, $enrolledcohort['visible']);
$this->assertEquals($cohort1->theme, $enrolledcohort['theme']);
> $this->assertIsArray($enrolledcohort['customfields']);
}
> $this->assertCount(2, $enrolledcohort['customfields']);
}
> $actual = [];
> foreach ($enrolledcohort['customfields'] as $customfield) {
// Check that a user with cohort:manage can see the cohort.
> $this->assertArrayHasKey('name', $customfield);
$this->unassignUserCapability('moodle/cohort:view', $context->id, $roleid);
> $this->assertArrayHasKey('shortname', $customfield);
$roleid = $this->assignUserCapability('moodle/cohort:manage', $context->id, $roleid);
> $this->assertArrayHasKey('type', $customfield);
// Call the external function.
> $this->assertArrayHasKey('valueraw', $customfield);
$returnedcohorts = core_cohort_external::get_cohorts(array(
> $this->assertArrayHasKey('value', $customfield);
$cohort1->id, $cohort2->id));
> $actual[$customfield['shortname']] = $customfield;
$returnedcohorts = \external_api::clean_returnvalue(core_cohort_external::get_cohorts_returns(), $returnedcohorts);
> }
> $this->assertEquals('Test value 1', $actual['testfield1']['value']);
// Check we retrieve the good total number of enrolled cohorts + no error on capability.
> $this->assertEquals('Test value 2', $actual['testfield2']['value']);
< $returnedcohorts = \external_api::clean_returnvalue(core_cohort_external::get_cohorts_returns(), $returnedcohorts);
> $returnedcohorts = external_api::clean_returnvalue(core_cohort_external::get_cohorts_returns(), $returnedcohorts);
// Check when allowcohortstheme is disabled, theme is not returned.
set_config('allowcohortthemes', 0);
$returnedcohorts = core_cohort_external::get_cohorts(array(
$cohort1->id));
< $returnedcohorts = \external_api::clean_returnvalue(core_cohort_external::get_cohorts_returns(), $returnedcohorts);
> $returnedcohorts = external_api::clean_returnvalue(core_cohort_external::get_cohorts_returns(), $returnedcohorts);
foreach ($returnedcohorts as $enrolledcohort) {
if ($enrolledcohort['idnumber'] == $cohort1->idnumber) {
$this->assertNull($enrolledcohort['theme']);
}
}
}
/**
* Test update_cohorts
*/
public function test_update_cohorts() {
< global $USER, $CFG, $DB;
> global $DB;
$this->resetAfterTest(true);
> // Custom fields.
set_config('allowcohortthemes', 0);
> $this->create_custom_fields();
>
$cohort1 = self::getDataGenerator()->create_cohort(array('visible' => 0));
> $data = cohort_handler::create()->export_instance_data_object($cohort1->id, true);
$cohort1 = array(
> $this->assertNull($data->testfield1);
'id' => $cohort1->id,
> $this->assertNull($data->testfield2);
'categorytype' => array('type' => 'id', 'value' => '1'),
>
'name' => 'cohortnametest1',
'idnumber' => 'idnumbertest1',
'description' => 'This is a description for cohort 1',
< 'theme' => 'classic'
> 'theme' => 'classic',
> 'customfields' => array(
> array(
> 'shortname' => 'testfield1',
> 'value' => 'Test value 1',
> ),
> array(
> 'shortname' => 'testfield2',
> 'value' => 'Test value 2',
> ),
> ),
);
$context = \context_system::instance();
$roleid = $this->assignUserCapability('moodle/cohort:manage', $context->id);
// Call the external function.
core_cohort_external::update_cohorts(array($cohort1));
$dbcohort = $DB->get_record('cohort', array('id' => $cohort1['id']));
$contextid = $DB->get_field('context', 'id', array('instanceid' => $cohort1['categorytype']['value'],
'contextlevel' => CONTEXT_COURSECAT));
$this->assertEquals($dbcohort->contextid, $contextid);
$this->assertEquals($dbcohort->name, $cohort1['name']);
$this->assertEquals($dbcohort->idnumber, $cohort1['idnumber']);
$this->assertEquals($dbcohort->description, $cohort1['description']);
$this->assertEquals($dbcohort->visible, 0);
$this->assertEmpty($dbcohort->theme);
> $data = cohort_handler::create()->export_instance_data_object($cohort1['id'], true);
> $this->assertEquals('Test value 1', $data->testfield1);
// Since field 'visible' was added in 2.8, make sure that update works correctly with and without this parameter.
> $this->assertEquals('Test value 2', $data->testfield2);
core_cohort_external::update_cohorts(array($cohort1 + array('visible' => 1)));
$dbcohort = $DB->get_record('cohort', array('id' => $cohort1['id']));
$this->assertEquals(1, $dbcohort->visible);
core_cohort_external::update_cohorts(array($cohort1));
$dbcohort = $DB->get_record('cohort', array('id' => $cohort1['id']));
$this->assertEquals(1, $dbcohort->visible);
// Call when $CFG->allowcohortthemes is enabled.
set_config('allowcohortthemes', 1);
core_cohort_external::update_cohorts(array($cohort1 + array('theme' => 'classic')));
$dbcohort = $DB->get_record('cohort', array('id' => $cohort1['id']));
$this->assertEquals('classic', $dbcohort->theme);
// Call when $CFG->allowcohortthemes is disabled.
set_config('allowcohortthemes', 0);
core_cohort_external::update_cohorts(array($cohort1 + array('theme' => 'boost')));
$dbcohort = $DB->get_record('cohort', array('id' => $cohort1['id']));
$this->assertEquals('classic', $dbcohort->theme);
> // Updating custom fields.
// Call without required capability.
> $cohort1['customfields'] = array(
$this->unassignUserCapability('moodle/cohort:manage', $context->id, $roleid);
> array(
$this->expectException(\required_capability_exception::class);
> 'shortname' => 'testfield1',
core_cohort_external::update_cohorts(array($cohort1));
> 'value' => 'Test value 1 updated',
}
> ),
> array(
/**
> 'shortname' => 'testfield2',
* Verify handling of 'id' param.
> 'value' => 'Test value 2 updated',
*/
> ),
public function test_update_cohorts_invalid_id_param() {
> );
$this->resetAfterTest(true);
> core_cohort_external::update_cohorts(array($cohort1));
$cohort = self::getDataGenerator()->create_cohort();
> $data = cohort_handler::create()->export_instance_data_object($cohort1['id'], true);
> $this->assertEquals('Test value 1 updated', $data->testfield1);
$cohort1 = array(
> $this->assertEquals('Test value 2 updated', $data->testfield2);
'id' => 'THIS IS NOT AN ID',
>
'name' => 'Changed cohort name',
'categorytype' => array('type' => 'id', 'value' => '1'),
'idnumber' => $cohort->idnumber,
);
try {
core_cohort_external::update_cohorts(array($cohort1));
$this->fail('Expecting invalid_parameter_exception exception, none occured');
} catch (\invalid_parameter_exception $e1) {
$this->assertStringContainsString('Invalid external api parameter: the value is "THIS IS NOT AN ID"', $e1->debuginfo);
}
$cohort1['id'] = 9.999; // Also not a valid id of a cohort.
try {
core_cohort_external::update_cohorts(array($cohort1));
$this->fail('Expecting invalid_parameter_exception exception, none occured');
} catch (\invalid_parameter_exception $e2) {
$this->assertStringContainsString('Invalid external api parameter: the value is "9.999"', $e2->debuginfo);
}
}
/**
* Test update_cohorts without permission on the dest category.
*/
public function test_update_cohorts_missing_dest() {
global $USER, $CFG, $DB;
$this->resetAfterTest(true);
$category1 = self::getDataGenerator()->create_category(array(
'name' => 'Test category 1'
));
$category2 = self::getDataGenerator()->create_category(array(
'name' => 'Test category 2'
));
$context1 = \context_coursecat::instance($category1->id);
$context2 = \context_coursecat::instance($category2->id);
$cohort = array(
'contextid' => $context1->id,
'name' => 'cohortnametest1',
'idnumber' => 'idnumbertest1',
'description' => 'This is a description for cohort 1'
);
$cohort1 = self::getDataGenerator()->create_cohort($cohort);
$roleid = $this->assignUserCapability('moodle/cohort:manage', $context1->id);
$cohortupdate = array(
'id' => $cohort1->id,
'categorytype' => array('type' => 'id', 'value' => $category2->id),
'name' => 'cohort update',
'idnumber' => 'idnumber update',
'description' => 'This is a description update'
);
// Call the external function.
// Should fail because we don't have permission on the dest category
$this->expectException(\required_capability_exception::class);
core_cohort_external::update_cohorts(array($cohortupdate));
}
/**
* Test update_cohorts without permission on the src category.
*/
public function test_update_cohorts_missing_src() {
global $USER, $CFG, $DB;
$this->resetAfterTest(true);
$category1 = self::getDataGenerator()->create_category(array(
'name' => 'Test category 1'
));
$category2 = self::getDataGenerator()->create_category(array(
'name' => 'Test category 2'
));
$context1 = \context_coursecat::instance($category1->id);
$context2 = \context_coursecat::instance($category2->id);
$cohort = array(
'contextid' => $context1->id,
'name' => 'cohortnametest1',
'idnumber' => 'idnumbertest1',
'description' => 'This is a description for cohort 1'
);
$cohort1 = self::getDataGenerator()->create_cohort($cohort);
$roleid = $this->assignUserCapability('moodle/cohort:manage', $context2->id);
$cohortupdate = array(
'id' => $cohort1->id,
'categorytype' => array('type' => 'id', 'value' => $category2->id),
'name' => 'cohort update',
'idnumber' => 'idnumber update',
'description' => 'This is a description update'
);
// Call the external function.
// Should fail because we don't have permission on the src category
$this->expectException(\required_capability_exception::class);
core_cohort_external::update_cohorts(array($cohortupdate));
}
/**
* Test add_cohort_members
*/
public function test_add_cohort_members() {
global $DB;
$this->resetAfterTest(true); // Reset all changes automatically after this test.
$contextid = \context_system::instance()->id;
$cohort = array(
'contextid' => $contextid,
'name' => 'cohortnametest1',
'idnumber' => 'idnumbertest1',
'description' => 'This is a description for cohort 1'
);
$cohort0 = self::getDataGenerator()->create_cohort($cohort);
// Check the cohorts were correctly created.
$this->assertEquals(1, $DB->count_records_select('cohort', ' (id = :cohortid0)',
array('cohortid0' => $cohort0->id)));
$cohort1 = array(
'cohorttype' => array('type' => 'id', 'value' => $cohort0->id),
'usertype' => array('type' => 'id', 'value' => '1')
);
$roleid = $this->assignUserCapability('moodle/cohort:assign', $contextid);
// Call the external function.
$addcohortmembers = core_cohort_external::add_cohort_members(array($cohort1));
< $addcohortmembers = \external_api::clean_returnvalue(core_cohort_external::add_cohort_members_returns(), $addcohortmembers);
> $addcohortmembers = external_api::clean_returnvalue(core_cohort_external::add_cohort_members_returns(), $addcohortmembers);
// Check we retrieve the good total number of created cohorts + no error on capability.
$this->assertEquals(1, count($addcohortmembers));
foreach ($addcohortmembers as $addcohortmember) {
$dbcohort = $DB->get_record('cohort_members', array('cohortid' => $cohort0->id));
$this->assertEquals($dbcohort->cohortid, $cohort1['cohorttype']['value']);
$this->assertEquals($dbcohort->userid, $cohort1['usertype']['value']);
}
// Call without required capability.
$cohort2 = array(
'cohorttype' => array('type' => 'id', 'value' => $cohort0->id),
'usertype' => array('type' => 'id', 'value' => '2')
);
$this->unassignUserCapability('moodle/cohort:assign', $contextid, $roleid);
$this->expectException(\required_capability_exception::class);
$addcohortmembers = core_cohort_external::add_cohort_members(array($cohort2));
}
/**
* Test delete_cohort_members
*/
public function test_delete_cohort_members() {
global $DB;
$this->resetAfterTest(true); // Reset all changes automatically after this test.
$cohort1 = self::getDataGenerator()->create_cohort();
$user1 = self::getDataGenerator()->create_user();
$cohort2 = self::getDataGenerator()->create_cohort();
$user2 = self::getDataGenerator()->create_user();
$context = \context_system::instance();
$roleid = $this->assignUserCapability('moodle/cohort:assign', $context->id);
$cohortaddmember1 = array(
'cohorttype' => array('type' => 'id', 'value' => $cohort1->id),
'usertype' => array('type' => 'id', 'value' => $user1->id)
);
$cohortmembers1 = core_cohort_external::add_cohort_members(array($cohortaddmember1));
< $cohortmembers1 = \external_api::clean_returnvalue(core_cohort_external::add_cohort_members_returns(), $cohortmembers1);
> $cohortmembers1 = external_api::clean_returnvalue(core_cohort_external::add_cohort_members_returns(), $cohortmembers1);
$cohortaddmember2 = array(
'cohorttype' => array('type' => 'id', 'value' => $cohort2->id),
'usertype' => array('type' => 'id', 'value' => $user2->id)
);
$cohortmembers2 = core_cohort_external::add_cohort_members(array($cohortaddmember2));
< $cohortmembers2 = \external_api::clean_returnvalue(core_cohort_external::add_cohort_members_returns(), $cohortmembers2);
> $cohortmembers2 = external_api::clean_returnvalue(core_cohort_external::add_cohort_members_returns(), $cohortmembers2);
// Check we retrieve no cohorts + no error on capability.
$this->assertEquals(2, $DB->count_records_select('cohort_members', ' ((cohortid = :idcohort1 AND userid = :iduser1)
OR (cohortid = :idcohort2 AND userid = :iduser2))',
array('idcohort1' => $cohort1->id, 'iduser1' => $user1->id, 'idcohort2' => $cohort2->id, 'iduser2' => $user2->id)));
// Call the external function.
$cohortdel1 = array(
'cohortid' => $cohort1->id,
'userid' => $user1->id
);
$cohortdel2 = array(
'cohortid' => $cohort2->id,
'userid' => $user2->id
);
core_cohort_external::delete_cohort_members(array($cohortdel1, $cohortdel2));
// Check we retrieve no cohorts + no error on capability.
$this->assertEquals(0, $DB->count_records_select('cohort_members', ' ((cohortid = :idcohort1 AND userid = :iduser1)
OR (cohortid = :idcohort2 AND userid = :iduser2))',
array('idcohort1' => $cohort1->id, 'iduser1' => $user1->id, 'idcohort2' => $cohort2->id, 'iduser2' => $user2->id)));
// Call without required capability.
$this->unassignUserCapability('moodle/cohort:assign', $context->id, $roleid);
$this->expectException(\required_capability_exception::class);
core_cohort_external::delete_cohort_members(array($cohortdel1, $cohortdel2));
}
/**
* Search cohorts.
*/
public function test_search_cohorts() {
global $DB, $CFG;
$this->resetAfterTest(true);
> $this->create_custom_fields();
$creator = $this->getDataGenerator()->create_user();
$user = $this->getDataGenerator()->create_user();
$catuser = $this->getDataGenerator()->create_user();
$catcreator = $this->getDataGenerator()->create_user();
$courseuser = $this->getDataGenerator()->create_user();
$category = $this->getDataGenerator()->create_category();
$othercategory = $this->getDataGenerator()->create_category();
$course = $this->getDataGenerator()->create_course();
$syscontext = \context_system::instance();
$catcontext = \context_coursecat::instance($category->id);
$coursecontext = \context_course::instance($course->id);
// Fetching default authenticated user role.
$authrole = $DB->get_record('role', array('id' => $CFG->defaultuserroleid));
// Reset all default authenticated users permissions.
unassign_capability('moodle/cohort:manage', $authrole->id);
// Creating specific roles.
$creatorrole = create_role('Creator role', 'creatorrole', 'creator role description');
$userrole = create_role('User role', 'userrole', 'user role description');
$courserole = create_role('Course user role', 'courserole', 'course user role description');
assign_capability('moodle/cohort:manage', CAP_ALLOW, $creatorrole, $syscontext->id);
assign_capability('moodle/cohort:view', CAP_ALLOW, $courserole, $syscontext->id);
// Check for parameter $includes = 'parents'.
role_assign($creatorrole, $creator->id, $syscontext->id);
role_assign($creatorrole, $catcreator->id, $catcontext->id);
role_assign($userrole, $user->id, $syscontext->id);
role_assign($userrole, $catuser->id, $catcontext->id);
// Enrol user in the course.
$this->getDataGenerator()->enrol_user($courseuser->id, $course->id, 'courserole');
$syscontext = array('contextid' => \context_system::instance()->id);
$catcontext = array('contextid' => \context_coursecat::instance($category->id)->id);
$othercatcontext = array('contextid' => \context_coursecat::instance($othercategory->id)->id);
$coursecontext = array('contextid' => \context_course::instance($course->id)->id);
> $customfields = array(
> 'contextid' => \context_system::instance()->id,
$cohort1 = $this->getDataGenerator()->create_cohort(array_merge($syscontext, array('name' => 'Cohortsearch 1')));
> 'customfield_testfield1' => 'Test value 1',
$cohort2 = $this->getDataGenerator()->create_cohort(array_merge($catcontext, array('name' => 'Cohortsearch 2')));
> 'customfield_testfield2' => 'Test value 2',
$cohort3 = $this->getDataGenerator()->create_cohort(array_merge($othercatcontext, array('name' => 'Cohortsearch 3')));
> );
>
// A user without permission in the system.
> // We need a site admin to be able to populate cohorts custom fields.
$this->setUser($user);
> $this->setAdminUser();
try {
> $cohort4 = $this->getDataGenerator()->create_cohort(array_merge($customfields, array('name' => 'Cohortsearch 4')));
$result = core_cohort_external::search_cohorts("Cohortsearch", $syscontext, 'parents');
$this->fail('Invalid permissions in system');
} catch (\required_capability_exception $e) {
// All good.
}
// A user without permission in a category.
$this->setUser($catuser);
try {
$result = core_cohort_external::search_cohorts("Cohortsearch", $catcontext, 'parents');
$this->fail('Invalid permissions in category');
} catch (\required_capability_exception $e) {
// All good.
}
// A user with permissions in the system.
$this->setUser($creator);
< $result = core_cohort_external::search_cohorts("Cohortsearch", $syscontext, 'parents');
< $this->assertEquals(1, count($result['cohorts']));
< $this->assertEquals('Cohortsearch 1', $result['cohorts'][$cohort1->id]->name);
> $result = core_cohort_external::search_cohorts("Cohortsearch 4", $syscontext, 'parents');
> $this->assertCount(1, $result['cohorts']);
> $this->assertEquals('Cohortsearch 4', $result['cohorts'][$cohort4->id]->name);
>
> $result = core_cohort_external::search_cohorts("Cohortsearch 4", $syscontext, 'parents');
> $this->assertCount(1, $result['cohorts']);
> $this->assertEquals('Cohortsearch 4', $result['cohorts'][$cohort4->id]->name);
> $this->assertIsArray($result['cohorts'][$cohort4->id]->customfields);
> $this->assertCount(2, $result['cohorts'][$cohort4->id]->customfields);
> $actual = [];
> foreach ($result['cohorts'][$cohort4->id]->customfields as $customfield) {
> $this->assertArrayHasKey('name', $customfield);
> $this->assertArrayHasKey('shortname', $customfield);
> $this->assertArrayHasKey('type', $customfield);
> $this->assertArrayHasKey('valueraw', $customfield);
> $this->assertArrayHasKey('value', $customfield);
> $actual[$customfield['shortname']] = $customfield;
> }
> $this->assertEquals('Test value 1', $actual['testfield1']['value']);
> $this->assertEquals('Test value 2', $actual['testfield2']['value']);
// A user with permissions in the category.
$this->setUser($catcreator);
$result = core_cohort_external::search_cohorts("Cohortsearch", $catcontext, 'parents');
< $this->assertEquals(2, count($result['cohorts']));
> $this->assertCount(3, $result['cohorts']);
$cohorts = array();
foreach ($result['cohorts'] as $cohort) {
$cohorts[] = $cohort->name;
}
$this->assertTrue(in_array('Cohortsearch 1', $cohorts));
// Check for parameter $includes = 'self'.
$this->setUser($creator);
$result = core_cohort_external::search_cohorts("Cohortsearch", $othercatcontext, 'self');
< $this->assertEquals(1, count($result['cohorts']));
> $this->assertCount(1, $result['cohorts']);
$this->assertEquals('Cohortsearch 3', $result['cohorts'][$cohort3->id]->name);
// Check for parameter $includes = 'all'.
$this->setUser($creator);
$result = core_cohort_external::search_cohorts("Cohortsearch", $syscontext, 'all');
< $this->assertEquals(3, count($result['cohorts']));
> $this->assertCount(4, $result['cohorts']);
// A user in the course context with the system cohort:view capability. Check that all the system cohorts are returned.
$this->setUser($courseuser);
$result = core_cohort_external::search_cohorts("Cohortsearch", $coursecontext, 'all');
< $this->assertEquals(1, count($result['cohorts']));
> $this->assertCount(2, $result['cohorts']);
$this->assertEquals('Cohortsearch 1', $result['cohorts'][$cohort1->id]->name);
// Detect invalid parameter $includes.
$this->setUser($creator);
try {
$result = core_cohort_external::search_cohorts("Cohortsearch", $syscontext, 'invalid');
$this->fail('Invalid parameter includes');
} catch (\coding_exception $e) {
// All good.
}
}
}