Ir a la última revisión | Autoría | Comparar con el anterior | Ultima modificación | Ver Log |
<?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_courseformat\local;use stdClass;/*** Section format actions class tests.** @package core_courseformat* @copyright 2023 Ferran Recio <ferran@moodle.com>* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later* @coversDefaultClass \core_courseformat\sectionactions*/class sectionactions_test extends \advanced_testcase {/*** Setup to ensure that fixtures are loaded.*/public static function setUpBeforeClass(): void {global $CFG;require_once($CFG->dirroot . '/course/lib.php');}/*** Test for create_delegated method.* @covers ::create_delegated* @dataProvider create_delegated_provider* @param string $component the name of the plugin* @param int|null $itemid the id of the delegated section* @param stdClass|null $fields the fields to set on the section*/public function test_create_delegated(string $component, ?int $itemid, ?stdClass $fields): void {global $DB;$this->resetAfterTest();$course = $this->getDataGenerator()->create_course(['format' => 'topics', 'numsections' => 1]);$sectionactions = new sectionactions($course);$section = $sectionactions->create_delegated($component, $itemid, $fields);$this->assertEquals($component, $section->component);$this->assertEquals($itemid, $section->itemid);if (!empty($fields)) {foreach ($fields as $field => $value) {$this->assertEquals($value, $section->$field);}}}/*** Data provider for test_create_delegated.* @return array*/public static function create_delegated_provider(): array {return ['component with no itemid or fields' => ['mod_assign',null,null,],'component with itemid but no fields' => ['mod_assign',1,null,],'component with itemid and empty fields' => ['mod_assign',1,new stdClass(),],'component with itemid and name field' => ['mod_assign',1,(object) ['name' => 'new name'],],'component with no itemid but name field' => ['mod_assign',null,(object) ['name' => 'new name'],],'component with itemid and summary' => ['mod_assign',1,(object) ['summary' => 'summary'],],'component with itemid and summary, summaryformat ' => ['mod_assign',1,(object) ['summary' => 'summary', 'summaryformat' => 1],],'component with itemid and section number' => ['mod_assign',1,(object) ['section' => 2],],'component with itemid and visible 1' => ['mod_assign',1,(object) ['visible' => 1],],'component with itemid and visible 0' => ['mod_assign',1,(object) ['visible' => 0],],];}/*** Test for create method.* @covers ::create* @dataProvider create_provider* @param int $sectionnum the name of the plugin* @param bool $skip if the validation should be skipped* @param bool $expectexception if the method should throw an exception* @param int $expected the expected section number*/public function test_create(int $sectionnum, bool $skip, bool $expectexception, int $expected): void {global $DB;$this->resetAfterTest();$course = $this->getDataGenerator()->create_course(['format' => 'topics', 'numsections' => 1]);$sectionactions = new sectionactions($course);if ($expectexception) {$this->expectException(\dml_write_exception::class);}$section = $sectionactions->create($sectionnum, $skip);$this->assertEquals($expected, $section->section);}/*** Data provider for test_create_delegated.* @return array*/public static function create_provider(): array {return ['section 1' => ['sectionnum' => 1,'skip' => false,'expectexception' => false,'expected' => 1,],'section 2' => ['sectionnum' => 2,'skip' => false,'expectexception' => false,'expected' => 2,],'section 3' => ['sectionnum' => 3,'skip' => false,'expectexception' => false,'expected' => 2,],'section 4' => ['sectionnum' => 4,'skip' => false,'expectexception' => false,'expected' => 2,],'section 1 with exception' => ['sectionnum' => 1,'skip' => true,'expectexception' => true,'expected' => 0,],'section 2 with skip validation' => ['sectionnum' => 2,'skip' => true,'expectexception' => false,'expected' => 2,],'section 5 with skip validation' => ['sectionnum' => 5,'skip' => true,'expectexception' => false,'expected' => 5,],];}/*** Test create sections when there are sections with comonent (delegated sections) in the course.* @covers ::create* @covers ::create_delegated*/public function test_create_with_delegated_sections(): void {global $DB;$this->resetAfterTest();$course = $this->getDataGenerator()->create_course(['format' => 'topics', 'numsections' => 1],['createsections' => true],);$sectionactions = new sectionactions($course);$section = $sectionactions->create_delegated('mod_forum', 1);$this->assertEquals(2, $section->section);$delegateid = $section->id;// Regular sections are created before delegated ones.$section = $sectionactions->create(2);$this->assertEquals(2, $section->section);$regularid = $section->id;$modinfo = get_fast_modinfo($course);$section2 = $modinfo->get_section_info(2);$this->assertEquals($regularid, $section2->id);$this->assertEquals(2, $section2->section);$sectiondelegated = $modinfo->get_section_info_by_component('mod_forum', 1);$this->assertEquals($delegateid, $sectiondelegated->id);$this->assertEquals(3, $sectiondelegated->section);// New delegates should be after the current delegate sections.$section = $sectionactions->create_delegated('mod_forum', 2);$this->assertEquals(4, $section->section);}/*** Test for create_if_missing method.* @covers ::create_if_missing* @dataProvider create_if_missing_provider* @param array $sectionnums the section numbers to create* @param bool $expected the expected result*/public function test_create_if_missing(array $sectionnums, bool $expected): void {global $DB;$this->resetAfterTest();$course = $this->getDataGenerator()->create_course(['format' => 'topics', 'numsections' => 2]);$sectionactions = new sectionactions($course);$result = $sectionactions->create_if_missing($sectionnums);$this->assertEquals($expected, $result);$modinfo = get_fast_modinfo($course);foreach ($sectionnums as $sectionnum) {$section = $modinfo->get_section_info($sectionnum);$this->assertEquals($sectionnum, $section->section);}}/*** Data provider for test_create_delegated.* @return array*/public static function create_if_missing_provider(): array {return ['existing section' => ['sectionnum' => [1],'expected' => false,],'unexisting section' => ['sectionnum' => [3],'expected' => true,],'several existing sections' => ['sectionnum' => [1, 2],'expected' => false,],'several unexisting sections' => ['sectionnum' => [3, 4],'expected' => true,],'empty array' => ['sectionnum' => [],'expected' => false,],'existent and unexistent sections' => ['sectionnum' => [1, 2, 3, 4],'expected' => true,],];}/*** Test create if missing when the course has delegated sections.* @covers ::create_if_missing* @covers ::create_delegated*/public function test_create_if_missing_with_delegated_sections(): void {global $DB;$this->resetAfterTest();$course = $this->getDataGenerator()->create_course(['format' => 'topics', 'numsections' => 1],['createsections' => true],);$sectionactions = new sectionactions($course);$section = $sectionactions->create_delegated('mod_forum', 1);$delegateid = $section->id;$result = $sectionactions->create_if_missing([1, 2]);$this->assertTrue($result);$modinfo = get_fast_modinfo($course);$section = $modinfo->get_section_info(2);$this->assertEquals(2, $section->section);$this->assertNotEquals($delegateid, $section->id);$delegatedsection = $modinfo->get_section_info_by_id($delegateid);$this->assertEquals(3, $delegatedsection->section);$result = $sectionactions->create_if_missing([1, 2]);$this->assertFalse($result);$result = $sectionactions->create_if_missing([1, 2, 3]);$this->assertTrue($result);$modinfo = get_fast_modinfo($course);$section = $modinfo->get_section_info(3);$this->assertEquals(3, $section->section);$this->assertNotEquals($delegateid, $section->id);$delegatedsection = $modinfo->get_section_info_by_id($delegateid);$this->assertEquals(4, $delegatedsection->section);$result = $sectionactions->create_if_missing([1, 2, 3]);$this->assertFalse($result);}/*** Test for delete method.* @covers ::delete*/public function test_delete(): void {global $DB;$this->resetAfterTest(true);$generator = $this->getDataGenerator();$course = $generator->create_course(['numsections' => 6, 'format' => 'topics'],['createsections' => true]);$assign0 = $generator->create_module('assign', ['course' => $course, 'section' => 0]);$assign1 = $generator->create_module('assign', ['course' => $course, 'section' => 1]);$assign21 = $generator->create_module('assign', ['course' => $course, 'section' => 2]);$assign22 = $generator->create_module('assign', ['course' => $course, 'section' => 2]);$assign3 = $generator->create_module('assign', ['course' => $course, 'section' => 3]);$assign5 = $generator->create_module('assign', ['course' => $course, 'section' => 5]);$assign6 = $generator->create_module('assign', ['course' => $course, 'section' => 6]);$this->setAdminUser();$sectionactions = new sectionactions($course);$sections = get_fast_modinfo($course)->get_section_info_all();// Attempt to delete 0-section.$this->assertFalse($sectionactions->delete($sections[0], true));$this->assertTrue($DB->record_exists('course_modules', ['id' => $assign0->cmid]));$this->assertEquals(6, course_get_format($course)->get_last_section_number());// Delete last section.$this->assertTrue($sectionactions->delete($sections[6], true));$this->assertFalse($DB->record_exists('course_modules', ['id' => $assign6->cmid]));$this->assertEquals(5, course_get_format($course)->get_last_section_number());// Delete empty section.$this->assertTrue($sectionactions->delete($sections[4], false));$this->assertEquals(4, course_get_format($course)->get_last_section_number());// Delete section in the middle (2).$this->assertFalse($sectionactions->delete($sections[2], false));$this->assertEquals(4, course_get_format($course)->get_last_section_number());$sections = get_fast_modinfo($course)->get_section_info_all();$this->assertTrue($sectionactions->delete($sections[2], true));$this->assertFalse($DB->record_exists('course_modules', ['id' => $assign21->cmid]));$this->assertFalse($DB->record_exists('course_modules', ['id' => $assign22->cmid]));$this->assertEquals(3, course_get_format($course)->get_last_section_number());$this->assertEquals([0 => [$assign0->cmid],1 => [$assign1->cmid],2 => [$assign3->cmid],3 => [$assign5->cmid],],get_fast_modinfo($course)->sections);// Remove marked section.course_set_marker($course->id, 1);$this->assertTrue(course_get_format($course)->is_section_current(1));$this->assertTrue($sectionactions->delete(get_fast_modinfo($course)->get_section_info(1),true));$this->assertFalse(course_get_format($course)->is_section_current(1));}/*** Test that triggering a course_section_deleted event works as expected.* @covers ::delete*/public function test_section_deleted_event(): void {global $USER, $DB;$this->resetAfterTest();$sink = $this->redirectEvents();// Create the course with sections.$course = $this->getDataGenerator()->create_course(['numsections' => 10], ['createsections' => true]);$coursecontext = \context_course::instance($course->id);$section = get_fast_modinfo($course)->get_section_info(10);$sectionrecord = $DB->get_record('course_sections', ['id' => $section->id]);$sectionactions = new sectionactions($course);$sectionactions->delete($section);$events = $sink->get_events();$event = array_pop($events); // Delete section event.$sink->close();// Validate event data.$this->assertInstanceOf('\core\event\course_section_deleted', $event);$this->assertEquals('course_sections', $event->objecttable);$this->assertEquals($section->id, $event->objectid);$this->assertEquals($course->id, $event->courseid);$this->assertEquals($coursecontext->id, $event->contextid);$this->assertEquals($section->section, $event->other['sectionnum']);$expecteddesc = "The user with id '{$event->userid}' deleted section number '{$event->other['sectionnum']}' " ."(section name '{$event->other['sectionname']}') for the course with id '{$event->courseid}'";$this->assertEquals($expecteddesc, $event->get_description());$this->assertEquals($sectionrecord, $event->get_record_snapshot('course_sections', $event->objectid));$this->assertNull($event->get_url());$this->assertEventContextNotUsed($event);}/*** Test async section deletion hook.* @covers ::delete*/public function test_async_section_deletion_hook_implemented(): void {// Async section deletion (provided section contains modules), depends on the 'true' being returned by at least one plugin// implementing the 'course_module_adhoc_deletion_recommended' hook. In core, is implemented by the course recyclebin,// which will only return true if the plugin is enabled. To make sure async deletion occurs, this test enables recyclebin.global $DB, $USER;$this->resetAfterTest(true);$this->setAdminUser();// Ensure recyclebin is enabled.set_config('coursebinenable', true, 'tool_recyclebin');// Create course, module and context.$generator = $this->getDataGenerator();$course = $generator->create_course(['numsections' => 4, 'format' => 'topics'], ['createsections' => true]);$assign0 = $generator->create_module('assign', ['course' => $course, 'section' => 2]);$assign1 = $generator->create_module('assign', ['course' => $course, 'section' => 2]);$assign2 = $generator->create_module('assign', ['course' => $course, 'section' => 2]);$assign3 = $generator->create_module('assign', ['course' => $course, 'section' => 0]);$sectionactions = new sectionactions($course);// Delete empty section. No difference from normal, synchronous behaviour.$this->assertTrue($sectionactions->delete(get_fast_modinfo($course)->get_section_info(4), false, true));$this->assertEquals(3, course_get_format($course)->get_last_section_number());// Delete a module in section 2 (using async). Need to verify this doesn't generate two tasks when we delete// the section in the next step.course_delete_module($assign2->cmid, true);// Confirm that the module is pending deletion in its current section.$section = $DB->get_record('course_sections', ['course' => $course->id, 'section' => '2']); // For event comparison.$this->assertEquals(true, $DB->record_exists('course_modules', ['id' => $assign2->cmid, 'deletioninprogress' => 1,'section' => $section->id]));// Non-empty section, no forcedelete, so no change.$this->assertFalse($sectionactions->delete(get_fast_modinfo($course)->get_section_info(2), false, true));$sink = $this->redirectEvents();$this->assertTrue($sectionactions->delete(get_fast_modinfo($course)->get_section_info(2), true, true));// Now, confirm that:// a) the section's modules have been flagged for deletion and moved to section 0 and;// b) the section has been deleted and;// c) course_section_deleted event has been fired. The course_module_deleted events will only fire once they have been// removed from section 0 via the adhoc task.// Modules should have been flagged for deletion and moved to section 0.$sectionid = $DB->get_field('course_sections', 'id', ['course' => $course->id, 'section' => 0]);$this->assertEquals(3,$DB->count_records('course_modules', ['section' => $sectionid, 'deletioninprogress' => 1]));// Confirm the section has been deleted.$this->assertEquals(2, course_get_format($course)->get_last_section_number());// Check event fired.$events = $sink->get_events();$event = array_pop($events);$sink->close();$this->assertInstanceOf('\core\event\course_section_deleted', $event);$this->assertEquals($section->id, $event->objectid);$this->assertEquals($USER->id, $event->userid);$this->assertEquals('course_sections', $event->objecttable);$this->assertEquals(null, $event->get_url());$this->assertEquals($section, $event->get_record_snapshot('course_sections', $section->id));// Now, run the adhoc task to delete the modules from section 0.$sink = $this->redirectEvents(); // To capture the events.\phpunit_util::run_all_adhoc_tasks();// Confirm the modules have been deleted.list($insql, $assignids) = $DB->get_in_or_equal([$assign0->cmid, $assign1->cmid, $assign2->cmid]);$cmcount = $DB->count_records_select('course_modules', 'id ' . $insql, $assignids);$this->assertEmpty($cmcount);// Confirm other modules in section 0 still remain.$this->assertEquals(1, $DB->count_records('course_modules', ['id' => $assign3->cmid]));// Confirm that events were generated for all 3 of the modules.$events = $sink->get_events();$sink->close();$count = 0;while (!empty($events)) {$event = array_pop($events);if ($event instanceof \core\event\course_module_deleted &&in_array($event->objectid, [$assign0->cmid, $assign1->cmid, $assign2->cmid])) {$count++;}}$this->assertEquals(3, $count);}/*** Test section update method.** @covers ::update* @dataProvider update_provider* @param string $fieldname the name of the field to update* @param int|string $value the value to set* @param int|string $expected the expected value after the update ('=' to specify the same value as original field)* @param bool $expectexception if the method should throw an exception*/public function test_update(string $fieldname,int|string $value,int|string $expected,bool $expectexception): void {global $DB;$this->resetAfterTest();$course = $this->getDataGenerator()->create_course(['format' => 'topics', 'numsections' => 1],['createsections' => true]);$section = get_fast_modinfo($course)->get_section_info(1);$sectionrecord = $DB->get_record('course_sections', ['id' => $section->id]);$this->assertNotEquals($value, $sectionrecord->$fieldname);$this->assertNotEquals($value, $section->$fieldname);if ($expectexception) {$this->expectException(\moodle_exception::class);}if ($expected === '=') {$expected = $section->$fieldname;}$sectionactions = new sectionactions($course);$sectionactions->update($section, [$fieldname => $value]);$sectionrecord = $DB->get_record('course_sections', ['id' => $section->id]);$this->assertEquals($expected, $sectionrecord->$fieldname);$section = get_fast_modinfo($course)->get_section_info(1);$this->assertEquals($expected, $section->$fieldname);}/*** Data provider for test_update.* @return array*/public static function update_provider(): array {return ['Id will not be updated' => ['fieldname' => 'id','value' => -1,'expected' => '=','expectexception' => false,],'Course will not be updated' => ['fieldname' => 'course','value' => -1,'expected' => '=','expectexception' => false,],'Section number will not be updated' => ['fieldname' => 'section','value' => -1,'expected' => '=','expectexception' => false,],'Sequence will be updated' => ['fieldname' => 'name','value' => 'new name','expected' => 'new name','expectexception' => false,],'Summary can be updated' => ['fieldname' => 'summary','value' => 'new summary','expected' => 'new summary','expectexception' => false,],'Visible can be updated' => ['fieldname' => 'visible','value' => 0,'expected' => 0,'expectexception' => false,],'component can be updated' => ['fieldname' => 'component','value' => 'mod_assign','expected' => 'mod_assign','expectexception' => false,],'itemid can be updated' => ['fieldname' => 'itemid','value' => 1,'expected' => 1,'expectexception' => false,],'Long names throws and exception' => ['fieldname' => 'name','value' => str_repeat('a', 256),'expected' => '=','expectexception' => true,],];}/*** Test section update method updating several values at once.** @covers ::update*/public function test_update_multiple_fields(): void {global $DB;$this->resetAfterTest();$course = $this->getDataGenerator()->create_course(['format' => 'topics', 'numsections' => 1],['createsections' => true]);$section = get_fast_modinfo($course)->get_section_info(1);$sectionrecord = $DB->get_record('course_sections', ['id' => $section->id]);$this->assertEquals(1, $sectionrecord->visible);$this->assertNull($section->name);$sectionactions = new sectionactions($course);$sectionactions->update($section, ['name' => 'New name', 'visible' => 0]);$sectionrecord = $DB->get_record('course_sections', ['id' => $section->id]);$this->assertEquals('New name', $sectionrecord->name);$this->assertEquals(0, $sectionrecord->visible);$section = get_fast_modinfo($course)->get_section_info(1);$this->assertEquals('New name', $section->name);$this->assertEquals(0, $section->visible);}/*** Test updating a section trigger a course section update log event.** @covers ::update*/public function test_course_section_updated_event(): void {$this->resetAfterTest();$course = $this->getDataGenerator()->create_course(['format' => 'topics', 'numsections' => 1],['createsections' => true]);$section = get_fast_modinfo($course)->get_section_info(1);$sink = $this->redirectEvents();$sectionactions = new sectionactions($course);$sectionactions->update($section, ['name' => 'New name', 'visible' => 0]);$events = $sink->get_events();$event = reset($events);// Check that the event data is valid.$this->assertInstanceOf('\core\event\course_section_updated', $event);$data = $event->get_data();$this->assertEquals(\context_course::instance($course->id), $event->get_context());$this->assertEquals($section->id, $data['objectid']);}/*** Test section update change the modified date.** @covers ::update*/public function test_update_time_modified(): void {global $DB;$this->resetAfterTest();// Create the course with sections.$course = $this->getDataGenerator()->create_course(['format' => 'topics', 'numsections' => 1],['createsections' => true]);$section = get_fast_modinfo($course)->get_section_info(1);$sectionrecord = $DB->get_record('course_sections', ['id' => $section->id]);$oldtimemodified = $sectionrecord->timemodified;$sectionactions = new sectionactions($course);// Ensuring that the section update occurs at a different timestamp.$this->waitForSecond();// The timemodified should only be updated if the section is actually updated.$result = $sectionactions->update($section, []);$this->assertFalse($result);$sectionrecord = $DB->get_record('course_sections', ['id' => $section->id]);$this->assertEquals($oldtimemodified, $sectionrecord->timemodified);// Now update something to prove timemodified changes.$result = $sectionactions->update($section, ['name' => 'New name']);$this->assertTrue($result);$sectionrecord = $DB->get_record('course_sections', ['id' => $section->id]);$this->assertGreaterThan($oldtimemodified, $sectionrecord->timemodified);}/*** Test section updating visibility will hide or show section activities.** @covers ::update*/public function test_update_hide_section_activities(): void {global $DB;$this->resetAfterTest();// Create 4 activities (visible, visible, hidden, hidden).$course = $this->getDataGenerator()->create_course(['format' => 'topics', 'numsections' => 1],['createsections' => true]);$activity1 = $this->getDataGenerator()->create_module('assign',['course' => $course->id, 'section' => 1]);$activity2 = $this->getDataGenerator()->create_module('assign',['course' => $course->id, 'section' => 1]);$activity3 = $this->getDataGenerator()->create_module('assign',['course' => $course->id, 'section' => 1, 'visible' => 0]);$activity4 = $this->getDataGenerator()->create_module('assign',['course' => $course->id, 'section' => 1, 'visible' => 0]);$modinfo = get_fast_modinfo($course);$cm1 = $modinfo->get_cm($activity1->cmid);$cm2 = $modinfo->get_cm($activity2->cmid);$cm3 = $modinfo->get_cm($activity3->cmid);$cm4 = $modinfo->get_cm($activity4->cmid);$this->assertEquals(1, $cm1->visible);$this->assertEquals(1, $cm2->visible);$this->assertEquals(0, $cm3->visible);$this->assertEquals(0, $cm4->visible);$sectionactions = new sectionactions($course);// Validate hidding section hides all activities.$section = $modinfo->get_section_info(1);$sectionactions->update($section, ['visible' => 0]);$modinfo = get_fast_modinfo($course);$cm1 = $modinfo->get_cm($activity1->cmid);$cm2 = $modinfo->get_cm($activity2->cmid);$cm3 = $modinfo->get_cm($activity3->cmid);$cm4 = $modinfo->get_cm($activity4->cmid);$this->assertEquals(0, $cm1->visible);$this->assertEquals(0, $cm2->visible);$this->assertEquals(0, $cm3->visible);$this->assertEquals(0, $cm4->visible);// Validate showing the section restores the previous visibility.$section = $modinfo->get_section_info(1);$sectionactions->update($section, ['visible' => 1]);$modinfo = get_fast_modinfo($course);$cm1 = $modinfo->get_cm($activity1->cmid);$cm2 = $modinfo->get_cm($activity2->cmid);$cm3 = $modinfo->get_cm($activity3->cmid);$cm4 = $modinfo->get_cm($activity4->cmid);$this->assertEquals(1, $cm1->visible);$this->assertEquals(1, $cm2->visible);$this->assertEquals(0, $cm3->visible);$this->assertEquals(0, $cm4->visible);// Swap two activities visibility to alter visible values.set_coursemodule_visible($cm2->id, 0, 0, true);set_coursemodule_visible($cm4->id, 1, 1, true);$modinfo = get_fast_modinfo($course);$cm1 = $modinfo->get_cm($activity1->cmid);$cm2 = $modinfo->get_cm($activity2->cmid);$cm3 = $modinfo->get_cm($activity3->cmid);$cm4 = $modinfo->get_cm($activity4->cmid);$this->assertEquals(1, $cm1->visible);$this->assertEquals(0, $cm2->visible);$this->assertEquals(0, $cm3->visible);$this->assertEquals(1, $cm4->visible);// Validate hidding the section again.$section = $modinfo->get_section_info(1);$sectionactions->update($section, ['visible' => 0]);$modinfo = get_fast_modinfo($course);$cm1 = $modinfo->get_cm($activity1->cmid);$cm2 = $modinfo->get_cm($activity2->cmid);$cm3 = $modinfo->get_cm($activity3->cmid);$cm4 = $modinfo->get_cm($activity4->cmid);$this->assertEquals(0, $cm1->visible);$this->assertEquals(0, $cm2->visible);$this->assertEquals(0, $cm3->visible);$this->assertEquals(0, $cm4->visible);// Validate showing the section once more to check previous state is restored.$section = $modinfo->get_section_info(1);$sectionactions->update($section, ['visible' => 1]);$modinfo = get_fast_modinfo($course);$cm1 = $modinfo->get_cm($activity1->cmid);$cm2 = $modinfo->get_cm($activity2->cmid);$cm3 = $modinfo->get_cm($activity3->cmid);$cm4 = $modinfo->get_cm($activity4->cmid);$this->assertEquals(1, $cm1->visible);$this->assertEquals(0, $cm2->visible);$this->assertEquals(0, $cm3->visible);$this->assertEquals(1, $cm4->visible);}/*** Test that the preprocess_section_name method can alter the section rename value.** @covers ::update* @covers ::preprocess_delegated_section_fields*/public function test_preprocess_section_name(): void {global $DB, $CFG;$this->resetAfterTest();require_once($CFG->libdir . '/tests/fixtures/sectiondelegatetest.php');$course = $this->getDataGenerator()->create_course();$sectionactions = new sectionactions($course);$section = $sectionactions->create_delegated('test_component', 1);$result = $sectionactions->update($section, ['name' => 'new_name']);$this->assertTrue($result);$section = $DB->get_record('course_sections', ['id' => $section->id]);$this->assertEquals('new_name_suffix', $section->name);$sectioninfo = get_fast_modinfo($course->id)->get_section_info_by_id($section->id);$this->assertEquals('new_name_suffix', $sectioninfo->name);// Validate null name.$section = $sectionactions->create_delegated('test_component', 1, (object)['name' => 'sample']);$result = $sectionactions->update($section, ['name' => null]);$this->assertTrue($result);$section = $DB->get_record('course_sections', ['id' => $section->id]);$this->assertEquals('null_name', $section->name);$sectioninfo = get_fast_modinfo($course->id)->get_section_info_by_id($section->id);$this->assertEquals('null_name', $sectioninfo->name);}}