<?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/>.
< /**
< * @package core_grades
< * @category phpunit
< * @copyright nicolas@moodle.com
< * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
< */
> namespace core;
defined('MOODLE_INTERNAL') || die();
require_once(__DIR__.'/fixtures/lib.php');
<
< class core_grade_grade_testcase extends grade_base_testcase {
> /**
> * Test grade grades
> *
> * @package core
> * @category test
> * @copyright nicolas@moodle.com
> * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
> */
> class grade_grade_test extends \grade_base_testcase {
public function test_grade_grade() {
$this->sub_test_grade_grade_construct();
$this->sub_test_grade_grade_insert();
$this->sub_test_grade_grade_update();
$this->sub_test_grade_grade_fetch();
$this->sub_test_grade_grade_fetch_all();
$this->sub_test_grade_grade_load_grade_item();
$this->sub_test_grade_grade_standardise_score();
$this->sub_test_grade_grade_is_locked();
$this->sub_test_grade_grade_set_hidden();
$this->sub_test_grade_grade_is_hidden();
$this->sub_test_grade_grade_deleted();
$this->sub_test_grade_grade_deleted_event();
}
protected function sub_test_grade_grade_construct() {
< $params = new stdClass();
> $params = new \stdClass();
$params->itemid = $this->grade_items[0]->id;
$params->userid = 1;
$params->rawgrade = 88;
$params->rawgrademax = 110;
$params->rawgrademin = 18;
< $grade_grade = new grade_grade($params, false);
> $grade_grade = new \grade_grade($params, false);
$this->assertEquals($params->itemid, $grade_grade->itemid);
$this->assertEquals($params->rawgrade, $grade_grade->rawgrade);
}
protected function sub_test_grade_grade_insert() {
< $grade_grade = new grade_grade();
> $grade_grade = new \grade_grade();
$this->assertTrue(method_exists($grade_grade, 'insert'));
$grade_grade->itemid = $this->grade_items[0]->id;
$grade_grade->userid = 10;
$grade_grade->rawgrade = 88;
$grade_grade->rawgrademax = 110;
$grade_grade->rawgrademin = 18;
// Check the grade_item's needsupdate variable first.
$grade_grade->load_grade_item();
$this->assertEmpty($grade_grade->grade_item->needsupdate);
$grade_grade->insert();
$last_grade_grade = end($this->grade_grades);
$this->assertEquals($grade_grade->id, $last_grade_grade->id + 1);
// Timecreated will only be set if the grade was submitted by an activity module.
$this->assertTrue(empty($grade_grade->timecreated));
// Timemodified will only be set if the grade was submitted by an activity module.
$this->assertTrue(empty($grade_grade->timemodified));
// Keep our collection the same as is in the database.
$this->grade_grades[] = $grade_grade;
}
protected function sub_test_grade_grade_update() {
< $grade_grade = new grade_grade($this->grade_grades[0], false);
> $grade_grade = new \grade_grade($this->grade_grades[0], false);
$this->assertTrue(method_exists($grade_grade, 'update'));
}
protected function sub_test_grade_grade_fetch() {
< $grade_grade = new grade_grade();
> $grade_grade = new \grade_grade();
$this->assertTrue(method_exists($grade_grade, 'fetch'));
< $grades = grade_grade::fetch(array('id'=>$this->grade_grades[0]->id));
> $grades = \grade_grade::fetch(array('id'=>$this->grade_grades[0]->id));
$this->assertEquals($this->grade_grades[0]->id, $grades->id);
$this->assertEquals($this->grade_grades[0]->rawgrade, $grades->rawgrade);
}
protected function sub_test_grade_grade_fetch_all() {
< $grade_grade = new grade_grade();
> $grade_grade = new \grade_grade();
$this->assertTrue(method_exists($grade_grade, 'fetch_all'));
< $grades = grade_grade::fetch_all(array());
> $grades = \grade_grade::fetch_all(array());
$this->assertEquals(count($this->grade_grades), count($grades));
}
protected function sub_test_grade_grade_load_grade_item() {
< $grade_grade = new grade_grade($this->grade_grades[0], false);
> $grade_grade = new \grade_grade($this->grade_grades[0], false);
$this->assertTrue(method_exists($grade_grade, 'load_grade_item'));
$this->assertNull($grade_grade->grade_item);
$this->assertNotEmpty($grade_grade->itemid);
$this->assertNotNull($grade_grade->load_grade_item());
$this->assertNotNull($grade_grade->grade_item);
$this->assertEquals($this->grade_items[0]->id, $grade_grade->grade_item->id);
}
protected function sub_test_grade_grade_standardise_score() {
< $this->assertEquals(4, round(grade_grade::standardise_score(6, 0, 7, 0, 5)));
< $this->assertEquals(40, grade_grade::standardise_score(50, 30, 80, 0, 100));
> $this->assertEquals(4, round(\grade_grade::standardise_score(6, 0, 7, 0, 5)));
> $this->assertEquals(40, \grade_grade::standardise_score(50, 30, 80, 0, 100));
}
/*
* Disabling this test: the set_locked() arguments have been modified, rendering these tests useless until they are re-written
protected function test_grade_grade_set_locked() {
< $grade_item = new grade_item($this->grade_items[0]);
< $grade = new grade_grade($grade_item->get_final(1));
> $grade_item = new \grade_item($this->grade_items[0]);
> $grade = new \grade_grade($grade_item->get_final(1));
$this->assertTrue(method_exists($grade, 'set_locked'));
$this->assertTrue(empty($grade_item->locked));
$this->assertTrue(empty($grade->locked));
$this->assertTrue($grade->set_locked(true));
$this->assertFalse(empty($grade->locked));
$this->assertTrue($grade->set_locked(false));
$this->assertTrue(empty($grade->locked));
$this->assertTrue($grade_item->set_locked(true, true));
< $grade = new grade_grade($grade_item->get_final(1));
> $grade = new \grade_grade($grade_item->get_final(1));
$this->assertFalse(empty($grade->locked));
$this->assertFalse($grade->set_locked(true, false));
$this->assertTrue($grade_item->set_locked(true, false));
< $grade = new grade_grade($grade_item->get_final(1));
> $grade = new \grade_grade($grade_item->get_final(1));
$this->assertTrue($grade->set_locked(true, false));
}
*/
protected function sub_test_grade_grade_is_locked() {
< $grade = new grade_grade($this->grade_grades[0], false);
> $grade = new \grade_grade($this->grade_grades[0], false);
$this->assertTrue(method_exists($grade, 'is_locked'));
$this->assertFalse($grade->is_locked());
$grade->locked = time();
$this->assertTrue($grade->is_locked());
}
protected function sub_test_grade_grade_set_hidden() {
< $grade = new grade_grade($this->grade_grades[0], false);
< $grade_item = new grade_item($this->grade_items[0], false);
> $grade = new \grade_grade($this->grade_grades[0], false);
> $grade_item = new \grade_item($this->grade_items[0], false);
$this->assertTrue(method_exists($grade, 'set_hidden'));
$this->assertEquals(0, $grade_item->hidden);
$this->assertEquals(0, $grade->hidden);
$grade->set_hidden(0);
$this->assertEquals(0, $grade->hidden);
$grade->set_hidden(1);
$this->assertEquals(1, $grade->hidden);
$grade->set_hidden(0);
$this->assertEquals(0, $grade->hidden);
}
protected function sub_test_grade_grade_is_hidden() {
< $grade = new grade_grade($this->grade_grades[0], false);
> $grade = new \grade_grade($this->grade_grades[0], false);
$this->assertTrue(method_exists($grade, 'is_hidden'));
$this->assertFalse($grade->is_hidden());
$grade->hidden = 1;
$this->assertTrue($grade->is_hidden());
$grade->hidden = time()-666;
$this->assertFalse($grade->is_hidden());
$grade->hidden = time()+666;
$this->assertTrue($grade->is_hidden());
}
> /**
public function test_flatten_dependencies() {
> * Test \grade_grade::flatten_dependencies_array()
// First test a simple normal case.
> *
$a = array(1 => array(2, 3), 2 => array(), 3 => array(4), 4 => array());
> * @covers \grade_grade::flatten_dependencies_array()
$b = array();
> */
$expecteda = array(1 => array(2, 3, 4), 2 => array(), 3 => array(4), 4 => array());
$expectedb = array(1 => 1);
< test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
> \test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
$this->assertSame($expecteda, $a);
$this->assertSame($expectedb, $b);
// Edge case - empty arrays.
$a = $b = $expecteda = $expectedb = array();
< test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
> \test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
$this->assertSame($expecteda, $a);
$this->assertSame($expectedb, $b);
// Circular dependency.
$a = array(1 => array(2), 2 => array(3), 3 => array(1));
$b = array();
$expecteda = array(1 => array(1, 2, 3), 2 => array(1, 2, 3), 3 => array(1, 2, 3));
< test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
> \test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
$this->assertSame($expecteda, $a);
// Note - we don't test the depth when we got circular dependencies - the main thing we wanted to test was that there was
// no ka-boom. The result would be hard to understand and doesn't matter.
// Circular dependency 2.
$a = array(1 => array(2), 2 => array(3), 3 => array(4), 4 => array(2, 1));
$b = array();
$expecteda = array(1 => array(1, 2, 3, 4), 2 => array(1, 2, 3, 4), 3 => array(1, 2, 3, 4), 4 => array(1, 2, 3, 4));
< test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
> \test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
$this->assertSame($expecteda, $a);
>
}
> // Missing first level dependency.
> $a = array(1 => array(2, 3), 3 => array(4), 4 => array());
public function test_grade_grade_min_max() {
> $b = array();
global $CFG;
> $expecteda = array(1 => array(2, 3, 4), 3 => array(4), 4 => array());
$initialminmaxtouse = $CFG->grade_minmaxtouse;
> $expectedb = array(1 => 1);
>
$this->setAdminUser();
> \test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
$course = $this->getDataGenerator()->create_course();
> $this->assertSame($expecteda, $a);
$user = $this->getDataGenerator()->create_user();
> $this->assertSame($expectedb, $b);
$assignrecord = $this->getDataGenerator()->create_module('assign', array('course' => $course, 'grade' => 100));
>
$cm = get_coursemodule_from_instance('assign', $assignrecord->id);
> // Missing 2nd level dependency.
$assigncontext = context_module::instance($cm->id);
> $a = array(1 => array(2, 3), 2 => array(), 3 => array(4));
$assign = new assign($assigncontext, $cm, $course);
> $b = array();
> $expecteda = array(1 => array(2, 3, 4), 2 => array(), 3 => array(4));
// Fetch the assignment item.
> $expectedb = array(1 => 1);
$giparams = array('itemtype' => 'mod', 'itemmodule' => 'assign', 'iteminstance' => $assignrecord->id,
>
'courseid' => $course->id, 'itemnumber' => 0);
> \test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
$gi = grade_item::fetch($giparams);
> $this->assertSame($expecteda, $a);
$this->assertEquals(0, $gi->grademin);
> $this->assertSame($expectedb, $b);
$this->assertEquals(100, $gi->grademax);
>
> // Null first level dependency.
// Give a grade to the student.
> $a = array(1 => array(2, null), 2 => array(3), 3 => array(4), 4 => array());
$usergrade = $assign->get_user_grade($user->id, true);
> $b = array();
$usergrade->grade = 10;
> $expecteda = array(1 => array(2, 3, 4), 2 => array(3, 4), 3 => array(4), 4 => array());
$assign->update_grade($usergrade);
> $expectedb = array(1 => 2, 2 => 1);
>
// Check the grade stored in gradebook.
> \test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
$gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $this->assertSame($expecteda, $a);
$this->assertEquals(10, $gg->rawgrade);
> $this->assertSame($expectedb, $b);
$this->assertEquals(0, $gg->get_grade_min());
>
$this->assertEquals(100, $gg->get_grade_max());
> // Null 2nd level dependency.
> $a = array(1 => array(2, 3), 2 => array(), 3 => array(4), 4 => array(null));
// Change the min/max grade of the item.
> $b = array();
$gi->grademax = 50;
> $expecteda = array(1 => array(2, 3, 4), 2 => array(), 3 => array(4), 4 => array());
$gi->grademin = 2;
> $expectedb = array(1 => 1);
$gi->update();
>
> \test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
// Fetch the updated item.
> $this->assertSame($expecteda, $a);
$gi = grade_item::fetch($giparams);
> $this->assertSame($expectedb, $b);
>
// Now check the grade grade min/max with system setting.
> // Straight null dependency.
$CFG->grade_minmaxtouse = GRADE_MIN_MAX_FROM_GRADE_ITEM;
> $a = array(1 => array(2, 3), 2 => array(), 3 => array(4), 4 => null);
grade_set_setting($course->id, 'minmaxtouse', null); // Ensure no course setting.
> $b = array();
> $expecteda = array(1 => array(2, 3, 4), 2 => array(), 3 => array(4), 4 => array());
$gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $expectedb = array(1 => 1);
$this->assertEquals(2, $gg->get_grade_min());
>
$this->assertEquals(50, $gg->get_grade_max());
> \test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
> $this->assertSame($expecteda, $a);
// Now with other system setting.
> $this->assertSame($expectedb, $b);
$CFG->grade_minmaxtouse = GRADE_MIN_MAX_FROM_GRADE_GRADE;
>
grade_set_setting($course->id, 'minmaxtouse', null); // Ensure no course setting, and reset static cache.
> // Also incorrect non-array dependency.
$gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $a = array(1 => array(2, 3), 2 => array(), 3 => array(4), 4 => 23);
$this->assertEquals(0, $gg->get_grade_min());
> $b = array();
$this->assertEquals(100, $gg->get_grade_max());
> $expecteda = array(1 => array(2, 3, 4), 2 => array(), 3 => array(4), 4 => array());
> $expectedb = array(1 => 1);
// Now with overriden setting in course.
>
$CFG->grade_minmaxtouse = GRADE_MIN_MAX_FROM_GRADE_ITEM;
> \test_grade_grade_flatten_dependencies_array::test_flatten_dependencies_array($a, $b);
grade_set_setting($course->id, 'minmaxtouse', GRADE_MIN_MAX_FROM_GRADE_GRADE);
> $this->assertSame($expecteda, $a);
$gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $this->assertSame($expectedb, $b);
< $assigncontext = context_module::instance($cm->id);
< $assign = new assign($assigncontext, $cm, $course);
> $assigncontext = \context_module::instance($cm->id);
> $assign = new \assign($assigncontext, $cm, $course);
< $gi = grade_item::fetch($giparams);
> $gi = \grade_item::fetch($giparams);
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
< $gi = grade_item::fetch($giparams);
> $gi = \grade_item::fetch($giparams);
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
$CFG->grade_minmaxtouse = $initialminmaxtouse;
}
public function test_grade_grade_min_max_with_course_item() {
global $CFG, $DB;
$initialminmaxtouse = $CFG->grade_minmaxtouse;
$this->setAdminUser();
$course = $this->getDataGenerator()->create_course();
$user = $this->getDataGenerator()->create_user();
< $gi = grade_item::fetch_course_item($course->id);
> $gi = \grade_item::fetch_course_item($course->id);
// Fetch the category item.
$this->assertEquals(0, $gi->grademin);
$this->assertEquals(100, $gi->grademax);
// Give a grade to the student.
$gi->update_final_grade($user->id, 10);
// Check the grade min/max stored in gradebook.
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
$this->assertEquals(0, $gg->get_grade_min());
$this->assertEquals(100, $gg->get_grade_max());
// Change the min/max grade of the item.
$gi->grademin = 2;
$gi->grademax = 50;
$gi->update();
// Fetch the updated item.
< $gi = grade_item::fetch_course_item($course->id);
> $gi = \grade_item::fetch_course_item($course->id);
// Now check the grade grade min/max with system setting.
$CFG->grade_minmaxtouse = GRADE_MIN_MAX_FROM_GRADE_ITEM;
grade_set_setting($course->id, 'minmaxtouse', null); // Ensure no course setting.
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
$this->assertEquals(0, $gg->get_grade_min());
$this->assertEquals(100, $gg->get_grade_max());
// Now with other system setting.
$CFG->grade_minmaxtouse = GRADE_MIN_MAX_FROM_GRADE_GRADE;
grade_set_setting($course->id, 'minmaxtouse', null); // Ensure no course setting, and reset static cache.
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
$this->assertEquals(0, $gg->get_grade_min());
$this->assertEquals(100, $gg->get_grade_max());
// Now with overriden setting in course.
$CFG->grade_minmaxtouse = GRADE_MIN_MAX_FROM_GRADE_ITEM;
grade_set_setting($course->id, 'minmaxtouse', GRADE_MIN_MAX_FROM_GRADE_GRADE);
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
$this->assertEquals(0, $gg->get_grade_min());
$this->assertEquals(100, $gg->get_grade_max());
$CFG->grade_minmaxtouse = GRADE_MIN_MAX_FROM_GRADE_GRADE;
grade_set_setting($course->id, 'minmaxtouse', GRADE_MIN_MAX_FROM_GRADE_ITEM);
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
$this->assertEquals(0, $gg->get_grade_min());
$this->assertEquals(100, $gg->get_grade_max());
$CFG->grade_minmaxtouse = $initialminmaxtouse;
}
public function test_grade_grade_min_max_with_category_item() {
global $CFG, $DB;
$initialminmaxtouse = $CFG->grade_minmaxtouse;
$this->setAdminUser();
$course = $this->getDataGenerator()->create_course();
$user = $this->getDataGenerator()->create_user();
< $coursegi = grade_item::fetch_course_item($course->id);
> $coursegi = \grade_item::fetch_course_item($course->id);
// Create a category item.
< $gc = new grade_category(array('courseid' => $course->id, 'fullname' => 'test'), false);
> $gc = new \grade_category(array('courseid' => $course->id, 'fullname' => 'test'), false);
$gc->insert();
$gi = $gc->get_grade_item();
$gi->grademax = 100;
$gi->grademin = 0;
$gi->update();
// Fetch the category item.
$giparams = array('itemtype' => 'category', 'iteminstance' => $gc->id);
< $gi = grade_item::fetch($giparams);
> $gi = \grade_item::fetch($giparams);
$this->assertEquals(0, $gi->grademin);
$this->assertEquals(100, $gi->grademax);
// Give a grade to the student.
$gi->update_final_grade($user->id, 10);
// Check the grade min/max stored in gradebook.
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
$this->assertEquals(0, $gg->get_grade_min());
$this->assertEquals(100, $gg->get_grade_max());
// Change the min/max grade of the item.
$gi->grademin = 2;
$gi->grademax = 50;
$gi->update();
// Fetch the updated item.
< $gi = grade_item::fetch($giparams);
> $gi = \grade_item::fetch($giparams);
// Now check the grade grade min/max with system setting.
$CFG->grade_minmaxtouse = GRADE_MIN_MAX_FROM_GRADE_ITEM;
grade_set_setting($course->id, 'minmaxtouse', null); // Ensure no course setting.
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
$this->assertEquals(0, $gg->get_grade_min());
$this->assertEquals(100, $gg->get_grade_max());
// Now with other system setting.
$CFG->grade_minmaxtouse = GRADE_MIN_MAX_FROM_GRADE_GRADE;
grade_set_setting($course->id, 'minmaxtouse', null); // Ensure no course setting, and reset static cache.
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
$this->assertEquals(0, $gg->get_grade_min());
$this->assertEquals(100, $gg->get_grade_max());
// Now with overriden setting in course.
$CFG->grade_minmaxtouse = GRADE_MIN_MAX_FROM_GRADE_ITEM;
grade_set_setting($course->id, 'minmaxtouse', GRADE_MIN_MAX_FROM_GRADE_GRADE);
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
$this->assertEquals(0, $gg->get_grade_min());
$this->assertEquals(100, $gg->get_grade_max());
$CFG->grade_minmaxtouse = GRADE_MIN_MAX_FROM_GRADE_GRADE;
grade_set_setting($course->id, 'minmaxtouse', GRADE_MIN_MAX_FROM_GRADE_ITEM);
< $gg = grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $user->id, 'itemid' => $gi->id));
$this->assertEquals(0, $gg->get_grade_min());
$this->assertEquals(100, $gg->get_grade_max());
$CFG->grade_minmaxtouse = $initialminmaxtouse;
}
/**
* Tests when a grade_grade has been deleted.
*/
public function sub_test_grade_grade_deleted() {
$dg = $this->getDataGenerator();
// Create the data we need for the tests.
< $fs = new file_storage();
> $fs = new \file_storage();
$u1 = $dg->create_user();
$c1 = $dg->create_course();
$a1 = $dg->create_module('assign', ['course' => $c1->id]);
< $a1context = context_module::instance($a1->cmid);
> $a1context = \context_module::instance($a1->cmid);
< $gi = new grade_item($dg->create_grade_item(
> $gi = new \grade_item($dg->create_grade_item(
[
'courseid' => $c1->id,
'itemtype' => 'mod',
'itemmodule' => 'assign',
'iteminstance' => $a1->id
]
), false);
// Add feedback files to copy as our update.
$this->add_feedback_file_to_copy();
$grades['feedback'] = 'Nice feedback!';
$grades['feedbackformat'] = FORMAT_MOODLE;
$grades['feedbackfiles'] = [
'contextid' => 1,
'component' => 'test',
'filearea' => 'testarea',
'itemid' => 1
];
$grades['userid'] = $u1->id;
grade_update('mod/assign', $gi->courseid, $gi->itemtype, $gi->itemmodule, $gi->iteminstance,
$gi->itemnumber, $grades);
// Feedback file area.
$files = $fs->get_area_files($a1context->id, GRADE_FILE_COMPONENT, GRADE_FEEDBACK_FILEAREA);
$this->assertEquals(2, count($files));
// History file area.
$files = $fs->get_area_files($a1context->id, GRADE_FILE_COMPONENT, GRADE_HISTORY_FEEDBACK_FILEAREA);
$this->assertEquals(2, count($files));
< $gg = grade_grade::fetch(array('userid' => $u1->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $u1->id, 'itemid' => $gi->id));
$gg->delete();
// Feedback file area.
$files = $fs->get_area_files($a1context->id, GRADE_FILE_COMPONENT, GRADE_FEEDBACK_FILEAREA);
$this->assertEquals(0, count($files));
// History file area.
$files = $fs->get_area_files($a1context->id, GRADE_FILE_COMPONENT, GRADE_HISTORY_FEEDBACK_FILEAREA);
$this->assertEquals(2, count($files));
}
/**
* Creates a feedback file to copy to the gradebook area.
*/
private function add_feedback_file_to_copy() {
$dummy = array(
'contextid' => 1,
'component' => 'test',
'filearea' => 'testarea',
'itemid' => 1,
'filepath' => '/',
'filename' => 'feedback1.txt'
);
$fs = get_file_storage();
$fs->create_file_from_string($dummy, '');
}
/**
* Tests grade_deleted event.
*/
public function sub_test_grade_grade_deleted_event() {
global $DB;
$dg = $this->getDataGenerator();
// Create the data we need for the tests.
$u1 = $dg->create_user();
$u2 = $dg->create_user();
$c1 = $dg->create_course();
$a1 = $dg->create_module('assign', ['course' => $c1->id]);
< $gi = new grade_item($dg->create_grade_item(
> $gi = new \grade_item($dg->create_grade_item(
[
'courseid' => $c1->id,
'itemtype' => 'mod',
'itemmodule' => 'assign',
'iteminstance' => $a1->id
]
), false);
grade_update('mod/assign', $gi->courseid, $gi->itemtype, $gi->itemmodule, $gi->iteminstance,
$gi->itemnumber, ['userid' => $u1->id]);
grade_update('mod/assign', $gi->courseid, $gi->itemtype, $gi->itemmodule, $gi->iteminstance,
$gi->itemnumber, ['userid' => $u2->id]);
< $gg = grade_grade::fetch(array('userid' => $u1->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $u1->id, 'itemid' => $gi->id));
$this->assertEquals($u1->id, $gg->userid);
$gg->load_grade_item();
$this->assertEquals($gi->id, $gg->grade_item->id);
// Delete user with valid grade item.
$sink = $this->redirectEvents();
grade_user_delete($u1->id);
$events = $sink->get_events();
$event = reset($events);
$sink->close();
$this->assertInstanceOf('core\event\grade_deleted', $event);
< $gg = grade_grade::fetch(array('userid' => $u2->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $u2->id, 'itemid' => $gi->id));
$this->assertEquals($u2->id, $gg->userid);
$gg->load_grade_item();
$this->assertEquals($gi->id, $gg->grade_item->id);
// Delete grade item, mock up orphaned grade_grades.
$DB->delete_records('grade_items', ['id' => $gi->id]);
< $gg = grade_grade::fetch(array('userid' => $u2->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $u2->id, 'itemid' => $gi->id));
$this->assertEquals($u2->id, $gg->userid);
// No event is triggered and there is a debugging message.
$sink = $this->redirectEvents();
grade_user_delete($u2->id);
$this->assertDebuggingCalled("Missing grade item id $gi->id");
$events = $sink->get_events();
$sink->close();
$this->assertEmpty($events);
// The grade should be deleted.
< $gg = grade_grade::fetch(array('userid' => $u2->id, 'itemid' => $gi->id));
> $gg = \grade_grade::fetch(array('userid' => $u2->id, 'itemid' => $gi->id));
$this->assertEmpty($gg);
}
/**
* Tests get_hiding_affected by locked category and overridden grades.
*/
public function test_category_get_hiding_affected() {
$generator = $this->getDataGenerator();
// Create the data we need for the tests.
$course1 = $generator->create_course();
$user1 = $generator->create_and_enrol($course1, 'student');
$assignment2 = $generator->create_module('assign', ['course' => $course1->id]);
// Create a category item.
< $gradecategory = new grade_category(array('courseid' => $course1->id, 'fullname' => 'test'), false);
> $gradecategory = new \grade_category(array('courseid' => $course1->id, 'fullname' => 'test'), false);
$gradecategoryid = $gradecategory->insert();
// Create one hidden grade item.
< $gradeitem1a = new grade_item($generator->create_grade_item(
> $gradeitem1a = new \grade_item($generator->create_grade_item(
[
'courseid' => $course1->id,
'itemtype' => 'mod',
'itemmodule' => 'assign',
'iteminstance' => $assignment2->id,
'categoryid' => $gradecategoryid,
'hidden' => 1,
]
), false);
grade_update('mod/assign', $gradeitem1a->courseid, $gradeitem1a->itemtype, $gradeitem1a->itemmodule, $gradeitem1a->iteminstance,
$gradeitem1a->itemnumber, ['userid' => $user1->id]);
// Get category grade item.
$gradeitem = $gradecategory->get_grade_item();
// Reset needsupdate to allow set_locked.
$gradeitem->needsupdate = 0;
$gradeitem->update();
// Lock category grade item.
$gradeitem->set_locked(1);
$hidingaffectedlocked = $this->call_get_hiding_affected($course1, $user1);
// Since locked category now should be recalculated.
// The number of unknown items is 2, this includes category item and course item.
$this->assertEquals(2, count($hidingaffectedlocked['unknown']));
// Unlock category.
$gradeitem->set_locked(0);
$hidingaffectedunlocked = $this->call_get_hiding_affected($course1, $user1);
// When category unlocked, hidden item should exist in altered items.
$this->assertTrue(in_array($gradeitem1a->id, array_keys($hidingaffectedunlocked['altered'])));
// This creates all the grade_grades we need.
grade_regrade_final_grades($course1->id);
// Set grade override.
< $gradegrade = grade_grade::fetch([
> $gradegrade = \grade_grade::fetch([
'userid' => $user1->id,
'itemid' => $gradeitem->id,
]);
// Set override grade grade, and check that grade submission has been overridden.
$gradegrade->set_overridden(true);
$this->assertEquals(true, $gradegrade->is_overridden());
$hidingaffectedoverridden = $this->call_get_hiding_affected($course1, $user1);
// No need to recalculate overridden grades.
$this->assertTrue(in_array($gradegrade->itemid, array_keys($hidingaffectedoverridden['alteredaggregationstatus'])));
$this->assertEquals('used', $hidingaffectedoverridden['alteredaggregationstatus'][$gradegrade->itemid]);
}
/**
* Call get_hiding_affected().
< * @param stdClass $course The course object
< * @param stdClass $user The student object
> * @param \stdClass $course The course object
> * @param \stdClass $user The student object
* @return array
*/
private function call_get_hiding_affected($course, $user) {
global $DB;
< $items = grade_item::fetch_all(array('courseid' => $course->id));
> $items = \grade_item::fetch_all(array('courseid' => $course->id));
$grades = array();
$sql = "SELECT g.*
FROM {grade_grades} g
JOIN {grade_items} gi ON gi.id = g.itemid
WHERE g.userid = :userid AND gi.courseid = :courseid";
if ($gradesrecords = $DB->get_records_sql($sql, ['userid' => $user->id, 'courseid' => $course->id])) {
foreach ($gradesrecords as $grade) {
< $grades[$grade->itemid] = new grade_grade($grade, false);
> $grades[$grade->itemid] = new \grade_grade($grade, false);
}
unset($gradesrecords);
}
foreach ($items as $itemid => $gradeitem) {
if (!isset($grades[$itemid])) {
< $gradegrade = new grade_grade();
> $gradegrade = new \grade_grade();
$gradegrade->userid = $user->id;
$gradegrade->itemid = $gradeitem->id;
$grades[$itemid] = $gradegrade;
}
$gradeitem->grade_item = $gradeitem;
}
< return grade_grade::get_hiding_affected($grades, $items);
> return \grade_grade::get_hiding_affected($grades, $items);
}
}