See Release Notes
Long Term Support Release
<?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/>. declare(strict_types=1); namespace core_course\reportbuilder\datasource; use completion_completion;> use completion_criteria_self;use core_reportbuilder\local\filters\boolean_select; use core_reportbuilder\local\filters\date;> use core_reportbuilder\local\filters\duration;use core_reportbuilder\local\filters\select;> use core_reportbuilder\local\filters\text;use core_reportbuilder_generator; use core_reportbuilder_testcase;> use core_user;use grade_item; defined('MOODLE_INTERNAL') || die(); global $CFG; require_once("{$CFG->dirroot}/reportbuilder/tests/helpers.php");< require_once("{$CFG->libdir}/gradelib.php");/** * Course participants datasource tests * * @package core_course * @covers \core_course\reportbuilder\datasource\participants< * @covers \core_course\reportbuilder\local\formatters\completion < * @covers \core_course\reportbuilder\local\formatters\enrolment* @copyright 2022 David Matamoros <davidmc@moodle.com> * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class participants_test extends core_reportbuilder_testcase { /**< * Test participants datasource> * Load required test libraries > */ > public static function setUpBeforeClass(): void { > global $CFG; > > require_once("{$CFG->libdir}/gradelib.php"); > require_once("{$CFG->dirroot}/completion/criteria/completion_criteria_self.php"); > } > > /** > * Test default datasource > */ > public function test_datasource_default(): void { > global $DB; > > $this->resetAfterTest(); > > // Course one, two manually enrolled users. > $courseone = $this->getDataGenerator()->create_course(['fullname' => 'Zebras']); > $userone = $this->getDataGenerator()->create_and_enrol($courseone, 'student', ['firstname' => 'Zoe']); > $usertwo = $this->getDataGenerator()->create_and_enrol($courseone, 'student', ['firstname' => 'Amy']); > > // Course two, two self enrolled users (one inactive). > $coursetwo = $this->getDataGenerator()->create_course(['fullname' => 'Aardvarks']); > > $enrol = $DB->get_record('enrol', ['courseid' => $coursetwo->id, 'enrol' => 'self']); > enrol_get_plugin($enrol->enrol)->update_status($enrol, ENROL_INSTANCE_ENABLED); > > $this->getDataGenerator()->enrol_user($userone->id, $coursetwo->id, null, 'self'); > $this->getDataGenerator()->enrol_user($usertwo->id, $coursetwo->id, null, 'self', 0, 0, ENROL_USER_SUSPENDED); > > /** @var core_reportbuilder_generator $generator */ > $generator = $this->getDataGenerator()->get_plugin_generator('core_reportbuilder'); > $report = $generator->create_report(['name' => 'Participants', 'source' => participants::class, 'default' => 1]); > > $content = $this->get_custom_report_content($report->get('id')); > > // Default columns are course, user, method. Sorted by each. > $courseoneurl = course_get_url($courseone); > $coursetwourl = course_get_url($coursetwo); > > $useroneurl = core_user::get_profile_url($userone); > $usertwourl = core_user::get_profile_url($usertwo); > > $this->assertEquals([ > ["<a href=\"{$coursetwourl}\">{$coursetwo->fullname}</a>", > "<a href=\"{$useroneurl}\">" . fullname($userone) . "</a>", 'Self enrolment (Student)'], > ["<a href=\"{$courseoneurl}\">{$courseone->fullname}</a>", > "<a href=\"{$usertwourl}\">" . fullname($usertwo) . "</a>", 'Manual enrolments'], > ["<a href=\"{$courseoneurl}\">{$courseone->fullname}</a>", > "<a href=\"{$useroneurl}\">" . fullname($userone) . "</a>", 'Manual enrolments'], > ], array_map('array_values', $content)); > } > > /** > * Test datasource columns that aren't added by default*/< public function test_participants_datasource(): void {> public function test_datasource_non_default_columns(): void {global $DB; $this->resetAfterTest(); $timestart = time() - DAYSECS; $timeend = $timestart + 3 * DAYSECS; $timecompleted = $timestart + 2 * DAYSECS; $timelastaccess = time() + 4 * DAYSECS; $category = $this->getDataGenerator()->create_category(['name' => 'Music']); $course = $this->getDataGenerator()->create_course([ 'category' => $category->id, 'fullname' => 'All about Lionel at the work place', 'enablecompletion' => true, 'startdate' => $timestart, 'enddate' => $timeend, ]); $user1 = self::getDataGenerator()->create_user(); $this->getDataGenerator()->enrol_user($user1->id, $course->id, 'student', 'manual', $timestart, $timeend, ENROL_USER_ACTIVE); // Add them to a group. $group = self::getDataGenerator()->create_group(['courseid' => $course->id]); self::getDataGenerator()->create_group_member(['groupid' => $group->id, 'userid' => $user1->id]);< // Mark course as completed for the user. < $ccompletion = new completion_completion(array('course' => $course->id, 'userid' => $user1->id));> // Create self completion, mark as complete for the user. > $criteriaconfig = (object) ['id' => $course->id, 'criteria_self' => true]; > (new completion_criteria_self())->update_config($criteriaconfig); > > $ccompletion = new completion_completion(['course' => $course->id, 'userid' => $user1->id]);$ccompletion->mark_enrolled($timestart); $ccompletion->mark_complete($timecompleted); // Update final grade for the user. $courseitem = grade_item::fetch_course_item($course->id); $courseitem->update_final_grade($user1->id, 42.5); // Set some last access value for the user in the course. $DB->insert_record('user_lastaccess', ['userid' => $user1->id, 'courseid' => $course->id, 'timeaccess' => $timelastaccess]); /** @var core_reportbuilder_generator $generator */ $generator = $this->getDataGenerator()->get_plugin_generator('core_reportbuilder');< $report = $generator->create_report(['name' => 'Courses', 'source' => participants::class, 'default' => false]);> $report = $generator->create_report(['name' => 'Participants', 'source' => participants::class, 'default' => false]);$generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'course:fullname']); $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'course_category:name']); $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'user:fullname']);< // Order by enrolment method. < $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'enrolment:method', 'sortenabled' => 1]);> > // Enrol entity (report ordering by enrolment name). > $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'enrol:name', 'sortenabled' => 1]); > $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'enrol:plugin']); > $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'enrol:enabled']); > $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'enrol:period']); > $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'enrol:startdate']); > $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'enrol:enddate']); > > // Role entity. > $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'role:name']); > $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'role:shortname']); > $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'role:description']); >$generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'group:name']);> $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'completion:criteria']);$generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'completion:completed']); $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'access:timeaccess']); $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'completion:progresspercent']); $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'completion:timeenrolled']); $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'completion:timestarted']); $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'completion:timecompleted']); $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'completion:reaggregate']); $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'completion:dayscourse']); $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'completion:daysuntilcompletion']); $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'completion:grade']); // Add filter to the report.< $generator->create_filter(['reportid' => $report->get('id'), 'uniqueidentifier' => 'enrolment:method']);> $generator->create_filter(['reportid' => $report->get('id'), 'uniqueidentifier' => 'enrol:plugin']);$content = $this->get_custom_report_content($report->get('id')); // It should get 3 records (manual enrolment, self and guest). $this->assertCount(3, $content); // Filter by Manual enrolment method. $content = $this->get_custom_report_content($report->get('id'), 30, [< 'enrolment:method_operator' => select::EQUAL_TO, < 'enrolment:method_value' => 'manual',> 'enrol:plugin_operator' => select::EQUAL_TO, > 'enrol:plugin_value' => 'manual',]); $this->assertCount(1, $content); $this->assertEquals([ 'All about Lionel at the work place', // Course name. 'Music', // Course category name. fullname($user1), // User fullname. 'Manual enrolments', // Enrolment method.> 'Manual enrolments', // Enrolment plugin. $group->name, // Group name. > 'Yes', // Enrolment enabled. 'Yes', // Course completed. > '', // Enrolment period. userdate($timelastaccess), // Time last access. > '', // Enrolment start date. '100.0%', // Progress percentage. > '', // Enrolment end date. userdate($timestart), // Time enrolled. > 'Student', // Role name. '', // Time started. > 'student', // Role shortname. userdate($timecompleted), // Time completed. > 'Students generally have fewer privileges within a course.', // Role description.'', // Reagreggate.> "All criteria below are required<ul>\n<li>Self completion: Self completion</li>\n</ul>", // Completion criteria.< '2', // Days taking course. < '2', // Days until completion.> 2, // Days taking course. > 2, // Days until completion.'42.50', // Grade. ], array_values($content[0])); } /**< * Data provider for {@see test_report_filters}> * Data provider for {@see test_datasource_filters}* * @return array */< public function filters_data_provider(): array {> public function datasource_filters_provider(): array { > global $DB; >return [ [ 'enrolment:status', [ 'enrolment:status_operator' => select::EQUAL_TO, 'enrolment:status_value' => 1, ],< 'Luna'> ['Luna'],], [ 'enrolment:timecreated', [ 'enrolment:timecreated_operator' => date::DATE_CURRENT, 'enrolment:timecreated_unit' => date::DATE_UNIT_DAY, ],< 'Kira'> ['Kira'],], [ 'enrolment:timestarted', [ 'enrolment:timestarted_operator' => date::DATE_CURRENT, 'enrolment:timecreated_unit' => date::DATE_UNIT_DAY, ],< 'Luna'> ['Luna'],], [ 'enrolment:timeended', [ 'enrolment:timeended_operator' => date::DATE_CURRENT, 'enrolment:timeended_unit' => date::DATE_UNIT_DAY, ],< 'Luna'> ['Luna'], > ], > [ > 'enrol:enabled', > [ > 'completion:enabled_operator' => boolean_select::CHECKED, > ], > ['Lionel', 'Kira', 'Luna'], > ], > [ > 'enrol:period', > [ > 'enrol:period_operator' => duration::DURATION_MAXIMUM, > 'enrol:period_unit' => MINSECS, > 'enrol:period_value' => 2, > ], > ['Lionel', 'Kira', 'Luna'], > ], > [ > 'enrol:startdate', > [ > 'enrol:startdate_operator' => date::DATE_EMPTY, > ], > ['Lionel', 'Kira', 'Luna'], > ], > [ > 'enrol:enddate', > [ > 'enrol:enddate_operator' => date::DATE_EMPTY, > ], > ['Lionel', 'Kira', 'Luna'], > ], > [ > 'enrol:customname', > [ > 'enrol:customname_operator' => text::IS_EMPTY, > ], > ['Luna', 'Kira', 'Lionel'], > ], > [ > 'enrol:customname', > [ > 'enrol:customname_operator' => text::IS_EQUAL_TO, > 'enrol:customname_value' => 'All night long' > ], > [], > ], > [ > 'role:name', > [ > 'role:name_operator' => select::EQUAL_TO, > 'role:name_value' => $DB->get_field('role', 'id', ['shortname' => 'editingteacher']), > ], > ['Luna'],], [ 'completion:completed', [ 'completion:completed_operator' => boolean_select::CHECKED,< 'completion:completed_unit' => 1,],< 'Lionel'> ['Lionel'],], [ 'completion:timecompleted', [ 'completion:timecompleted_operator' => date::DATE_NOT_EMPTY, ],< 'Lionel'> ['Lionel'], > ], > [ > 'completion:timeenrolled', > [ > 'completion:timeenrolled_operator' => date::DATE_NOT_EMPTY, > ], > ['Lionel'], > ], > [ > 'completion:timestarted', > [ > 'completion:timestarted_operator' => date::DATE_NOT_EMPTY, > ], > ['Lionel'], > ], > [ > 'completion:reaggregate', > [ > 'completion:reaggregate_operator' => date::DATE_NOT_EMPTY, > ], > ['Lionel'],], ]; } /** * Test getting filter SQL * * @param string $filter * @param array $filtervalues< * @param string $expected> * @param string[] $expected*< * @dataProvider filters_data_provider> * @dataProvider datasource_filters_provider*/< public function test_report_filters(string $filter, array $filtervalues, string $expected): void {> public function test_datasource_filters(string $filter, array $filtervalues, array $expected): void {global $DB; $this->resetAfterTest(); $timestart = time() - DAYSECS; $timeend = $timestart + 3 * DAYSECS; $timecompleted = $timestart + 2 * DAYSECS; $timelastaccess = time() + 4 * DAYSECS; $category = $this->getDataGenerator()->create_category(['name' => 'Music']); $course = $this->getDataGenerator()->create_course([ 'category' => $category->id, 'fullname' => 'All about Lionel at the work place', 'enablecompletion' => true, 'startdate' => $timestart, 'enddate' => $timeend, ]); $user1 = self::getDataGenerator()->create_user(['firstname' => 'Lionel']); $user2 = self::getDataGenerator()->create_user(['firstname' => 'Kira']); $user3 = self::getDataGenerator()->create_user(['firstname' => 'Luna']); $this->getDataGenerator()->enrol_user($user1->id, $course->id, 'student', 'manual', $timestart - 8 * DAYSECS, $timeend, ENROL_USER_ACTIVE); $this->getDataGenerator()->enrol_user($user2->id, $course->id, 'student', 'manual', $timestart, $timeend, ENROL_USER_ACTIVE);< $this->getDataGenerator()->enrol_user($user3->id, $course->id, 'student',> $this->getDataGenerator()->enrol_user($user3->id, $course->id, 'editingteacher','manual', time(), time(), ENROL_USER_SUSPENDED); // Mark course as completed for the user. $ccompletion = new completion_completion(array('course' => $course->id, 'userid' => $user1->id)); $ccompletion->mark_enrolled($timestart);> $ccompletion->mark_inprogress($timestart);$ccompletion->mark_complete($timecompleted); // Set some last access value for the user in the course. $DB->insert_record('user_lastaccess', ['userid' => $user1->id, 'courseid' => $course->id, 'timeaccess' => $timelastaccess]); /** @var core_reportbuilder_generator $generator */ $generator = $this->getDataGenerator()->get_plugin_generator('core_reportbuilder');< $report = $generator->create_report(['name' => 'Courses', 'source' => participants::class, 'default' => false]);> $report = $generator->create_report(['name' => 'Participants', 'source' => participants::class, 'default' => false]);// Add user firstname column to the report. $generator->create_column(['reportid' => $report->get('id'), 'uniqueidentifier' => 'user:firstname']); $DB->set_field('user_enrolments', 'timecreated', 0, ['userid' => $user1->id]); $DB->set_field('user_enrolments', 'timecreated', 0, ['userid' => $user3->id]); // Add filters to the report.< $generator->create_filter(['reportid' => $report->get('id'), 'uniqueidentifier' => 'enrolment:method']);> $generator->create_filter(['reportid' => $report->get('id'), 'uniqueidentifier' => 'enrol:plugin']);$generator->create_filter(['reportid' => $report->get('id'), 'uniqueidentifier' => $filter]); // Apply filters.< $filtermanual = ['enrolment:method_operator' => select::EQUAL_TO, 'enrolment:method_value' => 'manual'];> $filtermanual = ['enrol:plugin_operator' => select::EQUAL_TO, 'enrol:plugin_value' => 'manual'];$content = $this->get_custom_report_content($report->get('id'), 30, $filtermanual + $filtervalues);< $this->assertCount(1, $content); < $this->assertEquals($expected, $content[0]['c0_firstname']);> $this->assertEqualsCanonicalizing($expected, array_column($content, 'c0_firstname'));} /** * Stress test datasource * * In order to execute this test PHPUNIT_LONGTEST should be defined as true in phpunit.xml or directly in config.php */ public function test_stress_datasource(): void { if (!PHPUNIT_LONGTEST) { $this->markTestSkipped('PHPUNIT_LONGTEST is not defined'); } $this->resetAfterTest(); $course = $this->getDataGenerator()->create_course(); $this->getDataGenerator()->create_and_enrol($course); $this->datasource_stress_test_columns(participants::class); $this->datasource_stress_test_columns_aggregation(participants::class); $this->datasource_stress_test_conditions(participants::class, 'course:idnumber'); } }