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/>. /** * Quiz events tests. * * @package mod_quiz * @category phpunit * @copyright 2013 Adrian Greeve * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ namespace mod_quiz\event;< use quiz; < use quiz_attempt;> use mod_quiz\quiz_attempt; > use mod_quiz\quiz_settings;use context_module;< defined('MOODLE_INTERNAL') || die(); < < global $CFG; < require_once($CFG->dirroot . '/mod/quiz/attemptlib.php'); </** * Unit tests for quiz events. * * @package mod_quiz * @category phpunit * @copyright 2013 Adrian Greeve * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class events_test extends \advanced_testcase { /** * Setup a quiz. *< * @return quiz the generated quiz.> * @return quiz_settings the generated quiz.*/ protected function prepare_quiz() { $this->resetAfterTest(true); // Create a course $course = $this->getDataGenerator()->create_course(); // Make a quiz. $quizgenerator = $this->getDataGenerator()->get_plugin_generator('mod_quiz');< $quiz = $quizgenerator->create_instance(array('course' => $course->id, 'questionsperpage' => 0, < 'grade' => 100.0, 'sumgrades' => 2));> $quiz = $quizgenerator->create_instance(['course' => $course->id, 'questionsperpage' => 0, > 'grade' => 100.0, 'sumgrades' => 2]);$cm = get_coursemodule_from_instance('quiz', $quiz->id, $course->id); // Create a couple of questions. $questiongenerator = $this->getDataGenerator()->get_plugin_generator('core_question'); $cat = $questiongenerator->create_question_category();< $saq = $questiongenerator->create_question('shortanswer', null, array('category' => $cat->id)); < $numq = $questiongenerator->create_question('numerical', null, array('category' => $cat->id));> $saq = $questiongenerator->create_question('shortanswer', null, ['category' => $cat->id]); > $numq = $questiongenerator->create_question('numerical', null, ['category' => $cat->id]);// Add them to the quiz. quiz_add_quiz_question($saq->id, $quiz); quiz_add_quiz_question($numq->id, $quiz); // Make a user to do the quiz. $user1 = $this->getDataGenerator()->create_user(); $this->setUser($user1);< return quiz::create($quiz->id, $user1->id);> return quiz_settings::create($quiz->id, $user1->id);} /** * Setup a quiz attempt at the quiz created by {@link prepare_quiz()}. *< * @param quiz $quizobj the generated quiz.> * @param \mod_quiz\quiz_settings $quizobj the generated quiz.* @param bool $ispreview Make the attempt a preview attempt when true. * @return array with three elements, array($quizobj, $quba, $attempt) */ protected function prepare_quiz_attempt($quizobj, $ispreview = false) { // Start the attempt. $quba = \question_engine::make_questions_usage_by_activity('mod_quiz', $quizobj->get_context()); $quba->set_preferred_behaviour($quizobj->get_quiz()->preferredbehaviour); $timenow = time(); $attempt = quiz_create_attempt($quizobj, 1, false, $timenow, $ispreview); quiz_start_new_attempt($quizobj, $quba, $attempt, 1, $timenow); quiz_attempt_save_started($quizobj, $quba, $attempt);< return array($quizobj, $quba, $attempt);> return [$quizobj, $quba, $attempt];} /** * Setup some convenience test data with a single attempt. * * @param bool $ispreview Make the attempt a preview attempt when true. * @return array with three elements, array($quizobj, $quba, $attempt) */ protected function prepare_quiz_data($ispreview = false) { $quizobj = $this->prepare_quiz(); return $this->prepare_quiz_attempt($quizobj, $ispreview); } public function test_attempt_submitted() { list($quizobj, $quba, $attempt) = $this->prepare_quiz_data(); $attemptobj = quiz_attempt::create($attempt->id); // Catch the event. $sink = $this->redirectEvents(); $timefinish = time(); $attemptobj->process_finish($timefinish, false); $events = $sink->get_events(); $sink->close(); // Validate the event. $this->assertCount(3, $events); $event = $events[2]; $this->assertInstanceOf('\mod_quiz\event\attempt_submitted', $event); $this->assertEquals('quiz_attempts', $event->objecttable); $this->assertEquals($quizobj->get_context(), $event->get_context()); $this->assertEquals($attempt->userid, $event->relateduserid); $this->assertEquals(null, $event->other['submitterid']); // Should be the user, but PHP Unit complains...< $this->assertEquals('quiz_attempt_submitted', $event->get_legacy_eventname()); < $legacydata = new \stdClass(); < $legacydata->component = 'mod_quiz'; < $legacydata->attemptid = (string) $attempt->id; < $legacydata->timestamp = $timefinish; < $legacydata->userid = $attempt->userid; < $legacydata->cmid = $quizobj->get_cmid(); < $legacydata->courseid = $quizobj->get_courseid(); < $legacydata->quizid = $quizobj->get_quizid(); < // Submitterid should be the user, but as we are in PHP Unit, CLI_SCRIPT is set to true which sets null in submitterid. < $legacydata->submitterid = null; < $legacydata->timefinish = $timefinish; < $this->assertEventLegacyData($legacydata, $event);$this->assertEventContextNotUsed($event); } public function test_attempt_becameoverdue() { list($quizobj, $quba, $attempt) = $this->prepare_quiz_data(); $attemptobj = quiz_attempt::create($attempt->id); // Catch the event. $sink = $this->redirectEvents(); $timefinish = time(); $attemptobj->process_going_overdue($timefinish, false); $events = $sink->get_events(); $sink->close(); $this->assertCount(1, $events); $event = $events[0]; $this->assertInstanceOf('\mod_quiz\event\attempt_becameoverdue', $event); $this->assertEquals('quiz_attempts', $event->objecttable); $this->assertEquals($quizobj->get_context(), $event->get_context()); $this->assertEquals($attempt->userid, $event->relateduserid); $this->assertNotEmpty($event->get_description()); // Submitterid should be the user, but as we are in PHP Unit, CLI_SCRIPT is set to true which sets null in submitterid. $this->assertEquals(null, $event->other['submitterid']);< $this->assertEquals('quiz_attempt_overdue', $event->get_legacy_eventname()); < $legacydata = new \stdClass(); < $legacydata->component = 'mod_quiz'; < $legacydata->attemptid = (string) $attempt->id; < $legacydata->timestamp = $timefinish; < $legacydata->userid = $attempt->userid; < $legacydata->cmid = $quizobj->get_cmid(); < $legacydata->courseid = $quizobj->get_courseid(); < $legacydata->quizid = $quizobj->get_quizid(); < $legacydata->submitterid = null; // Should be the user, but PHP Unit complains... < $this->assertEventLegacyData($legacydata, $event);$this->assertEventContextNotUsed($event); } public function test_attempt_abandoned() { list($quizobj, $quba, $attempt) = $this->prepare_quiz_data(); $attemptobj = quiz_attempt::create($attempt->id); // Catch the event. $sink = $this->redirectEvents(); $timefinish = time(); $attemptobj->process_abandon($timefinish, false); $events = $sink->get_events(); $sink->close(); $this->assertCount(1, $events); $event = $events[0]; $this->assertInstanceOf('\mod_quiz\event\attempt_abandoned', $event); $this->assertEquals('quiz_attempts', $event->objecttable); $this->assertEquals($quizobj->get_context(), $event->get_context()); $this->assertEquals($attempt->userid, $event->relateduserid); // Submitterid should be the user, but as we are in PHP Unit, CLI_SCRIPT is set to true which sets null in submitterid. $this->assertEquals(null, $event->other['submitterid']);< $this->assertEquals('quiz_attempt_abandoned', $event->get_legacy_eventname()); < $legacydata = new \stdClass(); < $legacydata->component = 'mod_quiz'; < $legacydata->attemptid = (string) $attempt->id; < $legacydata->timestamp = $timefinish; < $legacydata->userid = $attempt->userid; < $legacydata->cmid = $quizobj->get_cmid(); < $legacydata->courseid = $quizobj->get_courseid(); < $legacydata->quizid = $quizobj->get_quizid(); < $legacydata->submitterid = null; // Should be the user, but PHP Unit complains... < $this->assertEventLegacyData($legacydata, $event);$this->assertEventContextNotUsed($event); } public function test_attempt_started() { $quizobj = $this->prepare_quiz(); $quba = \question_engine::make_questions_usage_by_activity('mod_quiz', $quizobj->get_context()); $quba->set_preferred_behaviour($quizobj->get_quiz()->preferredbehaviour); $timenow = time(); $attempt = quiz_create_attempt($quizobj, 1, false, $timenow); quiz_start_new_attempt($quizobj, $quba, $attempt, 1, $timenow); // Trigger and capture the event. $sink = $this->redirectEvents(); quiz_attempt_save_started($quizobj, $quba, $attempt); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\attempt_started', $event); $this->assertEquals('quiz_attempts', $event->objecttable); $this->assertEquals($attempt->id, $event->objectid); $this->assertEquals($attempt->userid, $event->relateduserid); $this->assertEquals($quizobj->get_context(), $event->get_context());< $this->assertEquals('quiz_attempt_started', $event->get_legacy_eventname());$this->assertEquals(\context_module::instance($quizobj->get_cmid()), $event->get_context());< // Check legacy log data. < $expected = array($quizobj->get_courseid(), 'quiz', 'attempt', 'review.php?attempt=' . $attempt->id, < $quizobj->get_quizid(), $quizobj->get_cmid()); < $this->assertEventLegacyLogData($expected, $event); < // Check legacy event data. < $legacydata = new \stdClass(); < $legacydata->component = 'mod_quiz'; < $legacydata->attemptid = $attempt->id; < $legacydata->timestart = $attempt->timestart; < $legacydata->timestamp = $attempt->timestart; < $legacydata->userid = $attempt->userid; < $legacydata->quizid = $quizobj->get_quizid(); < $legacydata->cmid = $quizobj->get_cmid(); < $legacydata->courseid = $quizobj->get_courseid(); < $this->assertEventLegacyData($legacydata, $event); < $this->assertEventContextNotUsed($event);} /** * Test the attempt question restarted event. * * There is no external API for replacing a question, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_attempt_question_restarted() { list($quizobj, $quba, $attempt) = $this->prepare_quiz_data(); $params = [ 'objectid' => 1, 'relateduserid' => 2, 'courseid' => $quizobj->get_courseid(), 'context' => \context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'page' => 2, 'slot' => 3, 'newquestionid' => 2 ] ]; $event = \mod_quiz\event\attempt_question_restarted::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\attempt_question_restarted', $event); $this->assertEquals(\context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the attempt updated event. * * There is no external API for updating an attempt, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_attempt_updated() { list($quizobj, $quba, $attempt) = $this->prepare_quiz_data(); $params = [ 'objectid' => 1, 'relateduserid' => 2, 'courseid' => $quizobj->get_courseid(), 'context' => \context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'page' => 0 ] ]; $event = \mod_quiz\event\attempt_updated::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\attempt_updated', $event); $this->assertEquals(\context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the attempt auto-saved event. * * There is no external API for auto-saving an attempt, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_attempt_autosaved() { list($quizobj, $quba, $attempt) = $this->prepare_quiz_data(); $params = [ 'objectid' => 1, 'relateduserid' => 2, 'courseid' => $quizobj->get_courseid(), 'context' => \context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'page' => 0 ] ]; $event = \mod_quiz\event\attempt_autosaved::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\attempt_autosaved', $event); $this->assertEquals(\context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the edit page viewed event. * * There is no external API for updating a quiz, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_edit_page_viewed() { $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);< $params = array(> $params = ['courseid' => $course->id, 'context' => \context_module::instance($quiz->cmid),< 'other' => array(> 'other' => ['quizid' => $quiz->id< ) < );> ] > ];$event = \mod_quiz\event\edit_page_viewed::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\edit_page_viewed', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context());< $expected = array($course->id, 'quiz', 'editquestions', 'view.php?id=' . $quiz->cmid, $quiz->id, $quiz->cmid); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test the attempt deleted event. */ public function test_attempt_deleted() { list($quizobj, $quba, $attempt) = $this->prepare_quiz_data(); // Trigger and capture the event. $sink = $this->redirectEvents(); quiz_delete_attempt($attempt, $quizobj->get_quiz()); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\attempt_deleted', $event); $this->assertEquals(\context_module::instance($quizobj->get_cmid()), $event->get_context());< $expected = array($quizobj->get_courseid(), 'quiz', 'delete attempt', 'report.php?id=' . $quizobj->get_cmid(), < $attempt->id, $quizobj->get_cmid()); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test that preview attempt deletions are not logged. */ public function test_preview_attempt_deleted() { // Create quiz with preview attempt. list($quizobj, $quba, $previewattempt) = $this->prepare_quiz_data(true); // Delete a preview attempt, capturing events. $sink = $this->redirectEvents(); quiz_delete_attempt($previewattempt, $quizobj->get_quiz()); // Verify that no events were generated. $this->assertEmpty($sink->get_events()); } /** * Test the report viewed event. * * There is no external API for viewing reports, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_report_viewed() { $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);< $params = array(> $params = ['context' => $context = \context_module::instance($quiz->cmid),< 'other' => array(> 'other' => ['quizid' => $quiz->id, 'reportname' => 'overview'< ) < );> ] > ];$event = \mod_quiz\event\report_viewed::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\report_viewed', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context());< $expected = array($course->id, 'quiz', 'report', 'report.php?id=' . $quiz->cmid . '&mode=overview', < $quiz->id, $quiz->cmid); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test the attempt reviewed event. * * There is no external API for reviewing attempts, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_attempt_reviewed() { $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);< $params = array(> $params = ['objectid' => 1, 'relateduserid' => 2, 'courseid' => $course->id, 'context' => \context_module::instance($quiz->cmid),< 'other' => array(> 'other' => ['quizid' => $quiz->id< ) < );> ] > ];$event = \mod_quiz\event\attempt_reviewed::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\attempt_reviewed', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context());< $expected = array($course->id, 'quiz', 'review', 'review.php?attempt=1', $quiz->id, $quiz->cmid); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test the attempt summary viewed event. * * There is no external API for viewing the attempt summary, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_attempt_summary_viewed() { $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);< $params = array(> $params = ['objectid' => 1, 'relateduserid' => 2, 'courseid' => $course->id, 'context' => \context_module::instance($quiz->cmid),< 'other' => array(> 'other' => ['quizid' => $quiz->id< ) < );> ] > ];$event = \mod_quiz\event\attempt_summary_viewed::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\attempt_summary_viewed', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context());< $expected = array($course->id, 'quiz', 'view summary', 'summary.php?attempt=1', $quiz->id, $quiz->cmid); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test the user override created event. * * There is no external API for creating a user override, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_user_override_created() { $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);< $params = array(> $params = ['objectid' => 1, 'relateduserid' => 2, 'context' => \context_module::instance($quiz->cmid),< 'other' => array(> 'other' => ['quizid' => $quiz->id< ) < );> ] > ];$event = \mod_quiz\event\user_override_created::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\user_override_created', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the group override created event. * * There is no external API for creating a group override, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_group_override_created() { $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);< $params = array(> $params = ['objectid' => 1, 'context' => \context_module::instance($quiz->cmid),< 'other' => array(> 'other' => ['quizid' => $quiz->id, 'groupid' => 2< ) < );> ] > ];$event = \mod_quiz\event\group_override_created::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\group_override_created', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the user override updated event. * * There is no external API for updating a user override, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_user_override_updated() { $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);< $params = array(> $params = ['objectid' => 1, 'relateduserid' => 2, 'context' => \context_module::instance($quiz->cmid),< 'other' => array(> 'other' => ['quizid' => $quiz->id< ) < );> ] > ];$event = \mod_quiz\event\user_override_updated::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\user_override_updated', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context());< $expected = array($course->id, 'quiz', 'edit override', 'overrideedit.php?id=1', $quiz->id, $quiz->cmid); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test the group override updated event. * * There is no external API for updating a group override, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_group_override_updated() { $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);< $params = array(> $params = ['objectid' => 1, 'context' => \context_module::instance($quiz->cmid),< 'other' => array(> 'other' => ['quizid' => $quiz->id, 'groupid' => 2< ) < );> ] > ];$event = \mod_quiz\event\group_override_updated::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\group_override_updated', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context());< $expected = array($course->id, 'quiz', 'edit override', 'overrideedit.php?id=1', $quiz->id, $quiz->cmid); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test the user override deleted event. */ public function test_user_override_deleted() { global $DB; $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);// Create an override. $override = new \stdClass(); $override->quiz = $quiz->id; $override->userid = 2; $override->id = $DB->insert_record('quiz_overrides', $override); // Trigger and capture the event. $sink = $this->redirectEvents(); quiz_delete_override($quiz, $override->id); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\user_override_deleted', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context());< $expected = array($course->id, 'quiz', 'delete override', 'overrides.php?cmid=' . $quiz->cmid, $quiz->id, $quiz->cmid); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test the group override deleted event. */ public function test_group_override_deleted() { global $DB; $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);// Create an override. $override = new \stdClass(); $override->quiz = $quiz->id; $override->groupid = 2; $override->id = $DB->insert_record('quiz_overrides', $override); // Trigger and capture the event. $sink = $this->redirectEvents(); quiz_delete_override($quiz, $override->id); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\group_override_deleted', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context());< $expected = array($course->id, 'quiz', 'delete override', 'overrides.php?cmid=' . $quiz->cmid, $quiz->id, $quiz->cmid); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test the attempt viewed event. * * There is no external API for continuing an attempt, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_attempt_viewed() { $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);< $params = array(> $params = ['objectid' => 1, 'relateduserid' => 2, 'courseid' => $course->id, 'context' => \context_module::instance($quiz->cmid),< 'other' => array(> 'other' => ['quizid' => $quiz->id, 'page' => 0< ) < );> ] > ];$event = \mod_quiz\event\attempt_viewed::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\attempt_viewed', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context());< $expected = array($course->id, 'quiz', 'continue attempt', 'review.php?attempt=1', $quiz->id, $quiz->cmid); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test the attempt previewed event. */ public function test_attempt_preview_started() { $quizobj = $this->prepare_quiz(); $quba = \question_engine::make_questions_usage_by_activity('mod_quiz', $quizobj->get_context()); $quba->set_preferred_behaviour($quizobj->get_quiz()->preferredbehaviour); $timenow = time(); $attempt = quiz_create_attempt($quizobj, 1, false, $timenow, true); quiz_start_new_attempt($quizobj, $quba, $attempt, 1, $timenow); // Trigger and capture the event. $sink = $this->redirectEvents(); quiz_attempt_save_started($quizobj, $quba, $attempt); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\attempt_preview_started', $event); $this->assertEquals(\context_module::instance($quizobj->get_cmid()), $event->get_context());< $expected = array($quizobj->get_courseid(), 'quiz', 'preview', 'view.php?id=' . $quizobj->get_cmid(), < $quizobj->get_quizid(), $quizobj->get_cmid()); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test the question manually graded event. * * There is no external API for manually grading a question, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_question_manually_graded() { list($quizobj, $quba, $attempt) = $this->prepare_quiz_data();< $params = array(> $params = ['objectid' => 1, 'courseid' => $quizobj->get_courseid(), 'context' => \context_module::instance($quizobj->get_cmid()),< 'other' => array(> 'other' => ['quizid' => $quizobj->get_quizid(), 'attemptid' => 2, 'slot' => 3< ) < );> ] > ];$event = \mod_quiz\event\question_manually_graded::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\question_manually_graded', $event); $this->assertEquals(\context_module::instance($quizobj->get_cmid()), $event->get_context());< $expected = array($quizobj->get_courseid(), 'quiz', 'manualgrade', 'comment.php?attempt=2&slot=3', < $quizobj->get_quizid(), $quizobj->get_cmid()); < $this->assertEventLegacyLogData($expected, $event);$this->assertEventContextNotUsed($event); } /** * Test the attempt regraded event. * * There is no external API for regrading attempts, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_attempt_regraded() { $this->resetAfterTest(); $this->setAdminUser(); $course = $this->getDataGenerator()->create_course();< $quiz = $this->getDataGenerator()->create_module('quiz', array('course' => $course->id));> $quiz = $this->getDataGenerator()->create_module('quiz', ['course' => $course->id]);< $params = array(> $params = ['objectid' => 1, 'relateduserid' => 2, 'courseid' => $course->id, 'context' => \context_module::instance($quiz->cmid),< 'other' => array(> 'other' => ['quizid' => $quiz->id< ) < );> ] > ];$event = \mod_quiz\event\attempt_regraded::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\attempt_regraded', $event); $this->assertEquals(\context_module::instance($quiz->cmid), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the attempt notify manual graded event. * There is no external API for notification email when manual grading of user's attempt is completed, * so the unit test will simply create and trigger the event and ensure the event data is returned as expected. */ public function test_attempt_manual_grading_completed() { $this->resetAfterTest(); list($quizobj, $quba, $attempt) = $this->prepare_quiz_data(); $attemptobj = quiz_attempt::create($attempt->id); $params = [ 'objectid' => $attemptobj->get_attemptid(), 'relateduserid' => $attemptobj->get_userid(), 'courseid' => $attemptobj->get_course()->id, 'context' => \context_module::instance($attemptobj->get_cmid()), 'other' => [ 'quizid' => $attemptobj->get_quizid() ] ]; $event = \mod_quiz\event\attempt_manual_grading_completed::create($params); // Catch the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $sink->close(); // Validate the event. $this->assertCount(1, $events); $event = reset($events); $this->assertInstanceOf('\mod_quiz\event\attempt_manual_grading_completed', $event); $this->assertEquals('quiz_attempts', $event->objecttable); $this->assertEquals($quizobj->get_context(), $event->get_context()); $this->assertEquals($attempt->userid, $event->relateduserid); $this->assertNotEmpty($event->get_description()); $this->assertEventContextNotUsed($event); } /** * Test the page break created event. * * There is no external API for creating page break, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_page_break_created() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => 1, 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'slotnumber' => 3, ] ]; $event = \mod_quiz\event\page_break_created::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\page_break_created', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the page break deleted event. * * There is no external API for deleting page break, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_page_deleted_created() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => 1, 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'slotnumber' => 3, ] ]; $event = \mod_quiz\event\page_break_deleted::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\page_break_deleted', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the quiz grade updated event. * * There is no external API for updating quiz grade, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_quiz_grade_updated() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => $quizobj->get_quizid(), 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'oldgrade' => 1, 'newgrade' => 3, ] ]; $event = \mod_quiz\event\quiz_grade_updated::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\quiz_grade_updated', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the quiz re-paginated event. * * There is no external API for re-paginating quiz, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_quiz_repaginated() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => $quizobj->get_quizid(), 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'slotsperpage' => 3, ] ]; $event = \mod_quiz\event\quiz_repaginated::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\quiz_repaginated', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the section break created event. * * There is no external API for creating section break, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_section_break_created() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => 1, 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'firstslotid' => 1, 'firstslotnumber' => 2, 'title' => 'New title' ] ]; $event = \mod_quiz\event\section_break_created::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\section_break_created', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertStringContainsString($params['other']['title'], $event->get_description()); $this->assertEventContextNotUsed($event); } /** * Test the section break deleted event. * * There is no external API for deleting section break, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_section_break_deleted() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => 1, 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'firstslotid' => 1, 'firstslotnumber' => 2 ] ]; $event = \mod_quiz\event\section_break_deleted::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\section_break_deleted', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the section shuffle updated event. * * There is no external API for updating section shuffle, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_section_shuffle_updated() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => 1, 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'firstslotnumber' => 2, 'shuffle' => true ] ]; $event = \mod_quiz\event\section_shuffle_updated::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\section_shuffle_updated', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the section title updated event. * * There is no external API for updating section title, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_section_title_updated() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => 1, 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'firstslotid' => 1, 'firstslotnumber' => 2, 'newtitle' => 'New title' ] ]; $event = \mod_quiz\event\section_title_updated::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\section_title_updated', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertStringContainsString($params['other']['newtitle'], $event->get_description()); $this->assertEventContextNotUsed($event); } /** * Test the slot created event. * * There is no external API for creating slot, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_slot_created() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => 1, 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'slotnumber' => 1, 'page' => 1 ] ]; $event = \mod_quiz\event\slot_created::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\slot_created', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the slot deleted event. * * There is no external API for deleting slot, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_slot_deleted() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => 1, 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'slotnumber' => 1, ] ]; $event = \mod_quiz\event\slot_deleted::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\slot_deleted', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the slot mark updated event. * * There is no external API for updating slot mark, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_slot_mark_updated() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => 1, 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'previousmaxmark' => 1, 'newmaxmark' => 2, ] ]; $event = \mod_quiz\event\slot_mark_updated::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\slot_mark_updated', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the slot moved event. * * There is no external API for moving slot, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_slot_moved() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => 1, 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'previousslotnumber' => 1, 'afterslotnumber' => 2, 'page' => 1 ] ]; $event = \mod_quiz\event\slot_moved::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\slot_moved', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } /** * Test the slot require previous updated event. * * There is no external API for updating slot require previous option, so the unit test will simply * create and trigger the event and ensure the event data is returned as expected. */ public function test_slot_requireprevious_updated() { $quizobj = $this->prepare_quiz(); $params = [ 'objectid' => 1, 'context' => context_module::instance($quizobj->get_cmid()), 'other' => [ 'quizid' => $quizobj->get_quizid(), 'requireprevious' => true ] ]; $event = \mod_quiz\event\slot_requireprevious_updated::create($params); // Trigger and capture the event. $sink = $this->redirectEvents(); $event->trigger(); $events = $sink->get_events(); $event = reset($events); // Check that the event data is valid. $this->assertInstanceOf('\mod_quiz\event\slot_requireprevious_updated', $event); $this->assertEquals(context_module::instance($quizobj->get_cmid()), $event->get_context()); $this->assertEventContextNotUsed($event); } }