mirror of
https://github.com/moodle/moodle.git
synced 2025-01-17 21:49:15 +01:00
MDL-32203 completion: Fix issues with data object where clauses
This commit is contained in:
parent
aa753ac24f
commit
dbfcf440fa
@ -103,44 +103,43 @@ if ($form->is_cancelled()){
|
|||||||
|
|
||||||
// Handle aggregation methods
|
// Handle aggregation methods
|
||||||
// Overall aggregation
|
// Overall aggregation
|
||||||
$aggregation = new completion_aggregation();
|
$aggdata = array(
|
||||||
$aggregation->course = $data->id;
|
'course' => $data->id,
|
||||||
$aggregation->criteriatype = null;
|
'criteriatype' => null
|
||||||
|
);
|
||||||
|
$aggregation = new completion_aggregation($aggdata);
|
||||||
$aggregation->setMethod($data->overall_aggregation);
|
$aggregation->setMethod($data->overall_aggregation);
|
||||||
$aggregation->insert();
|
$aggregation->save();
|
||||||
|
|
||||||
// Activity aggregation
|
// Activity aggregation
|
||||||
if (empty($data->activity_aggregation)) {
|
if (empty($data->activity_aggregation)) {
|
||||||
$data->activity_aggregation = 0;
|
$data->activity_aggregation = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
$aggregation = new completion_aggregation();
|
$aggdata['criteriatype'] = COMPLETION_CRITERIA_TYPE_ACTIVITY;
|
||||||
$aggregation->course = $data->id;
|
$aggregation = new completion_aggregation($aggdata);
|
||||||
$aggregation->criteriatype = COMPLETION_CRITERIA_TYPE_ACTIVITY;
|
|
||||||
$aggregation->setMethod($data->activity_aggregation);
|
$aggregation->setMethod($data->activity_aggregation);
|
||||||
$aggregation->insert();
|
$aggregation->save();
|
||||||
|
|
||||||
// Course aggregation
|
// Course aggregation
|
||||||
if (empty($data->course_aggregation)) {
|
if (empty($data->course_aggregation)) {
|
||||||
$data->course_aggregation = 0;
|
$data->course_aggregation = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
$aggregation = new completion_aggregation();
|
$aggdata['criteriatype'] = COMPLETION_CRITERIA_TYPE_COURSE;
|
||||||
$aggregation->course = $data->id;
|
$aggregation = new completion_aggregation($aggdata);
|
||||||
$aggregation->criteriatype = COMPLETION_CRITERIA_TYPE_COURSE;
|
|
||||||
$aggregation->setMethod($data->course_aggregation);
|
$aggregation->setMethod($data->course_aggregation);
|
||||||
$aggregation->insert();
|
$aggregation->save();
|
||||||
|
|
||||||
// Role aggregation
|
// Role aggregation
|
||||||
if (empty($data->role_aggregation)) {
|
if (empty($data->role_aggregation)) {
|
||||||
$data->role_aggregation = 0;
|
$data->role_aggregation = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
$aggregation = new completion_aggregation();
|
$aggdata['criteriatype'] = COMPLETION_CRITERIA_TYPE_ROLE;
|
||||||
$aggregation->course = $data->id;
|
$aggregation = new completion_aggregation($aggdata);
|
||||||
$aggregation->criteriatype = COMPLETION_CRITERIA_TYPE_ROLE;
|
|
||||||
$aggregation->setMethod($data->role_aggregation);
|
$aggregation->setMethod($data->role_aggregation);
|
||||||
$aggregation->insert();
|
$aggregation->save();
|
||||||
|
|
||||||
// Update course total passing grade
|
// Update course total passing grade
|
||||||
if (!empty($data->criteria_grade)) {
|
if (!empty($data->criteria_grade)) {
|
||||||
@ -152,7 +151,10 @@ if ($form->is_cancelled()){
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
redirect($CFG->wwwroot."/course/view.php?id=$course->id", get_string('changessaved'));
|
add_to_log($course->id, 'course', 'completion updated', 'completion.php?id='.$course->id);
|
||||||
|
|
||||||
|
$url = new moodle_url('/course/view.php', array('id' => $course->id));
|
||||||
|
redirect($url);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -49,13 +49,16 @@ class completion_aggregation extends data_object {
|
|||||||
*/
|
*/
|
||||||
public $required_fields = array('id', 'course', 'criteriatype', 'method', 'value');
|
public $required_fields = array('id', 'course', 'criteriatype', 'method', 'value');
|
||||||
|
|
||||||
|
/* @var array Array of unique fields, used in where clauses */
|
||||||
|
public $unique_fields = array('course', 'criteriatype');
|
||||||
|
|
||||||
/* @var int Course id */
|
/* @var int Course id */
|
||||||
public $course;
|
public $course;
|
||||||
|
|
||||||
/* @var int Criteria type this aggregation method applies to, or NULL for overall course aggregation */
|
/* @var int Criteria type this aggregation method applies to, or NULL for overall course aggregation */
|
||||||
public $criteriatype;
|
public $criteriatype;
|
||||||
|
|
||||||
/* @var int Aggregation method (COMPLETION_AGGREGATION_* constant)*/
|
/* @var int Aggregation method (COMPLETION_AGGREGATION_* constant) */
|
||||||
public $method;
|
public $method;
|
||||||
|
|
||||||
/* @var mixed Method value */
|
/* @var mixed Method value */
|
||||||
@ -98,4 +101,19 @@ class completion_aggregation extends data_object {
|
|||||||
$this->method = COMPLETION_AGGREGATION_ALL;
|
$this->method = COMPLETION_AGGREGATION_ALL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Save aggregation method to database
|
||||||
|
*
|
||||||
|
* @access public
|
||||||
|
* @return boolean
|
||||||
|
*/
|
||||||
|
public function save() {
|
||||||
|
if ($this->id) {
|
||||||
|
return $this->update();
|
||||||
|
} else {
|
||||||
|
return $this->insert();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -111,7 +111,7 @@ class completion_completion extends data_object {
|
|||||||
$this->timeenrolled = $timeenrolled;
|
$this->timeenrolled = $timeenrolled;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->_save();
|
return $this->_save();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -137,7 +137,7 @@ class completion_completion extends data_object {
|
|||||||
$this->timestarted = $timestarted;
|
$this->timestarted = $timestarted;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->_save();
|
return $this->_save();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -165,24 +165,24 @@ class completion_completion extends data_object {
|
|||||||
$this->timecompleted = $timecomplete;
|
$this->timecompleted = $timecomplete;
|
||||||
|
|
||||||
// Save record
|
// Save record
|
||||||
$this->_save();
|
return $this->_save();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Save course completion status
|
* Save course completion status
|
||||||
*
|
*
|
||||||
* This method creates a course_completions record if none exists
|
* This method creates a course_completions record if none exists
|
||||||
|
* @access private
|
||||||
|
* @return bool
|
||||||
*/
|
*/
|
||||||
private function _save() {
|
private function _save() {
|
||||||
global $DB;
|
|
||||||
|
|
||||||
if ($this->timeenrolled === null) {
|
if ($this->timeenrolled === null) {
|
||||||
$this->timeenrolled = 0;
|
$this->timeenrolled = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Save record
|
// Save record
|
||||||
if ($this->id) {
|
if ($this->id) {
|
||||||
$this->update();
|
return $this->update();
|
||||||
} else {
|
} else {
|
||||||
// Make sure reaggregate field is not null
|
// Make sure reaggregate field is not null
|
||||||
if (!$this->reaggregate) {
|
if (!$this->reaggregate) {
|
||||||
@ -191,10 +191,10 @@ class completion_completion extends data_object {
|
|||||||
|
|
||||||
// Make sure timestarted is not null
|
// Make sure timestarted is not null
|
||||||
if (!$this->timestarted) {
|
if (!$this->timestarted) {
|
||||||
$this->timestarted = 0;
|
$this->timestarted = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->insert();
|
return $this->insert();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -167,11 +167,11 @@ abstract class completion_criteria extends data_object {
|
|||||||
public static function factory($params) {
|
public static function factory($params) {
|
||||||
global $CFG, $COMPLETION_CRITERIA_TYPES;
|
global $CFG, $COMPLETION_CRITERIA_TYPES;
|
||||||
|
|
||||||
if (!isset($params->criteriatype) || !isset($COMPLETION_CRITERIA_TYPES[$params->criteriatype])) {
|
if (!isset($params['criteriatype']) || !isset($COMPLETION_CRITERIA_TYPES[$params['criteriatype']])) {
|
||||||
error('invalidcriteriatype', 'completion');
|
error('invalidcriteriatype', 'completion');
|
||||||
}
|
}
|
||||||
|
|
||||||
$class = 'completion_criteria_'.$COMPLETION_CRITERIA_TYPES[$params->criteriatype];
|
$class = 'completion_criteria_'.$COMPLETION_CRITERIA_TYPES[$params['criteriatype']];
|
||||||
require_once($CFG->libdir.'/completion/'.$class.'.php');
|
require_once($CFG->libdir.'/completion/'.$class.'.php');
|
||||||
|
|
||||||
return new $class($params, false);
|
return new $class($params, false);
|
||||||
|
@ -235,8 +235,7 @@ class completion_criteria_activity extends completion_criteria {
|
|||||||
// Loop through completions, and mark as complete
|
// Loop through completions, and mark as complete
|
||||||
$rs = $DB->get_recordset_sql($sql);
|
$rs = $DB->get_recordset_sql($sql);
|
||||||
foreach ($rs as $record) {
|
foreach ($rs as $record) {
|
||||||
|
$completion = new completion_criteria_completion($record, DATA_OBJECT_FETCH_BY_KEY);
|
||||||
$completion = new completion_criteria_completion((array)$record);
|
|
||||||
$completion->mark_complete($record->timecompleted);
|
$completion->mark_complete($record->timecompleted);
|
||||||
}
|
}
|
||||||
$rs->close();
|
$rs->close();
|
||||||
|
@ -44,6 +44,9 @@ class completion_criteria_completion extends data_object {
|
|||||||
/* @var array Array of required table fields, must start with 'id'. */
|
/* @var array Array of required table fields, must start with 'id'. */
|
||||||
public $required_fields = array('id', 'userid', 'course', 'criteriaid', 'gradefinal', 'rpl', 'deleted', 'unenroled', 'timecompleted');
|
public $required_fields = array('id', 'userid', 'course', 'criteriaid', 'gradefinal', 'rpl', 'deleted', 'unenroled', 'timecompleted');
|
||||||
|
|
||||||
|
/* @var array Array of unique fields, used in where clauses */
|
||||||
|
public $unique_fields = array('userid', 'course', 'criteriaid');
|
||||||
|
|
||||||
/* @var int User ID */
|
/* @var int User ID */
|
||||||
public $userid;
|
public $userid;
|
||||||
|
|
||||||
|
@ -193,8 +193,8 @@ class completion_criteria_course extends completion_criteria {
|
|||||||
// Loop through completions, and mark as complete
|
// Loop through completions, and mark as complete
|
||||||
$rs = $DB->get_recordset_sql($sql);
|
$rs = $DB->get_recordset_sql($sql);
|
||||||
foreach ($rs as $record) {
|
foreach ($rs as $record) {
|
||||||
$completion = new completion_criteria_completion((array)$record);
|
$completion = new completion_criteria_completion($record, DATA_OBJECT_FETCH_BY_KEY);
|
||||||
$completion->mark_complete($record->timecompleted);
|
$completion->mark_complete($record['timecompleted']);
|
||||||
}
|
}
|
||||||
$rs->close();
|
$rs->close();
|
||||||
}
|
}
|
||||||
|
@ -179,8 +179,8 @@ class completion_criteria_date extends completion_criteria {
|
|||||||
// Loop through completions, and mark as complete
|
// Loop through completions, and mark as complete
|
||||||
$rs = $DB->get_recordset_sql($sql, array(time()));
|
$rs = $DB->get_recordset_sql($sql, array(time()));
|
||||||
foreach ($rs as $record) {
|
foreach ($rs as $record) {
|
||||||
$completion = new completion_criteria_completion((array)$record);
|
$completion = new completion_criteria_completion($record, DATA_OBJECT_FETCH_BY_KEY);
|
||||||
$completion->mark_complete($record->timeend);
|
$completion->mark_complete($record['timeend']);
|
||||||
}
|
}
|
||||||
$rs->close();
|
$rs->close();
|
||||||
}
|
}
|
||||||
|
@ -229,8 +229,7 @@ class completion_criteria_duration extends completion_criteria {
|
|||||||
$now = time();
|
$now = time();
|
||||||
$rs = $DB->get_recordset_sql($sql, array($now, $now));
|
$rs = $DB->get_recordset_sql($sql, array($now, $now));
|
||||||
foreach ($rs as $record) {
|
foreach ($rs as $record) {
|
||||||
|
$completion = new completion_criteria_completion($record, DATA_OBJECT_FETCH_BY_KEY);
|
||||||
$completion = new completion_criteria_completion((array)$record);
|
|
||||||
|
|
||||||
// Use time start if not 0, otherwise use timeenrolled
|
// Use time start if not 0, otherwise use timeenrolled
|
||||||
if ($record->otimestart) {
|
if ($record->otimestart) {
|
||||||
|
@ -215,8 +215,8 @@ class completion_criteria_grade extends completion_criteria {
|
|||||||
// Loop through completions, and mark as complete
|
// Loop through completions, and mark as complete
|
||||||
$rs = $DB->get_recordset_sql($sql);
|
$rs = $DB->get_recordset_sql($sql);
|
||||||
foreach ($rs as $record) {
|
foreach ($rs as $record) {
|
||||||
$completion = new completion_criteria_completion((array)$record);
|
$completion = new completion_criteria_completion($record, DATA_OBJECT_FETCH_BY_KEY);
|
||||||
$completion->mark_complete($record->timecompleted);
|
$completion->mark_complete($record['timecompleted']);
|
||||||
}
|
}
|
||||||
$rs->close();
|
$rs->close();
|
||||||
}
|
}
|
||||||
|
@ -317,7 +317,7 @@ function completion_cron_completions() {
|
|||||||
foreach ($completions as $params) {
|
foreach ($completions as $params) {
|
||||||
$timecompleted = max($timecompleted, $params->timecompleted);
|
$timecompleted = max($timecompleted, $params->timecompleted);
|
||||||
|
|
||||||
$completion = new completion_criteria_completion($params, false);
|
$completion = new completion_criteria_completion((array)$params, false);
|
||||||
|
|
||||||
// Handle aggregation special cases
|
// Handle aggregation special cases
|
||||||
if ($params->criteriatype == COMPLETION_CRITERIA_TYPE_ACTIVITY) {
|
if ($params->criteriatype == COMPLETION_CRITERIA_TYPE_ACTIVITY) {
|
||||||
|
@ -26,6 +26,14 @@
|
|||||||
|
|
||||||
defined('MOODLE_INTERNAL') || die();
|
defined('MOODLE_INTERNAL') || die();
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Trigger for the new data_object api.
|
||||||
|
*
|
||||||
|
* See data_object::__constructor
|
||||||
|
*/
|
||||||
|
define('DATA_OBJECT_FETCH_BY_KEY', 2);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A data abstraction object that holds methods and attributes
|
* A data abstraction object that holds methods and attributes
|
||||||
*
|
*
|
||||||
@ -50,32 +58,74 @@ abstract class data_object {
|
|||||||
*/
|
*/
|
||||||
public $optional_fields = array();
|
public $optional_fields = array();
|
||||||
|
|
||||||
|
/* @var Array of unique fields, used in where clauses and constructor */
|
||||||
|
public $unique_fields = array();
|
||||||
|
|
||||||
/* @var int The primary key */
|
/* @var int The primary key */
|
||||||
public $id;
|
public $id;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor. Optionally (and by default) attempts to fetch corresponding row from DB.
|
* Constructor. Optionally (and by default) attempts to fetch corresponding row from DB.
|
||||||
*
|
*
|
||||||
* @param array $params an array with required parameters for this data object.
|
* If $fetch is not false, there are a few different things that can happen:
|
||||||
* @param bool $fetch Whether to fetch corresponding row from DB or not,
|
* - true:
|
||||||
* optional fields might not be defined if false used
|
* load corresponding row from the database, using $params as the WHERE clause
|
||||||
|
*
|
||||||
|
* - DATA_OBJECT_FETCH_BY_KEY:
|
||||||
|
* load corresponding row from the database, using only the $id in the WHERE clause (if set),
|
||||||
|
* otherwise using the columns listed in $this->unique_fields.
|
||||||
|
*
|
||||||
|
* - array():
|
||||||
|
* load corresponding row from the database, using the columns listed in this array
|
||||||
|
* in the WHERE clause
|
||||||
|
*
|
||||||
|
* @param array $params required parameters and their values for this data object
|
||||||
|
* @param mixed $fetch if false, do not attempt to fetch from the database, otherwise see notes
|
||||||
*/
|
*/
|
||||||
public function __construct($params = null, $fetch = true) {
|
public function __construct($params = null, $fetch = true) {
|
||||||
if (!empty($params) and (is_array($params) or is_object($params))) {
|
|
||||||
if ($fetch) {
|
|
||||||
if ($data = $this->fetch($params)) {
|
|
||||||
self::set_properties($this, $data);
|
|
||||||
} else {
|
|
||||||
self::set_properties($this, $this->optional_fields);//apply defaults for optional fields
|
|
||||||
self::set_properties($this, $params);
|
|
||||||
}
|
|
||||||
|
|
||||||
} else {
|
if (is_object($params)) {
|
||||||
self::set_properties($this, $params);
|
throw new coding_exception('data_object params should be in the form of an array, not an object');
|
||||||
|
}
|
||||||
|
|
||||||
|
// If no params given, apply defaults for optional fields
|
||||||
|
if (empty($params) || !is_array($params)) {
|
||||||
|
self::set_properties($this, $this->optional_fields);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If fetch is false, do not load from database
|
||||||
|
if ($fetch === false) {
|
||||||
|
self::set_properties($this, $params);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Compose where clause only from fields in unique_fields
|
||||||
|
if ($fetch === DATA_OBJECT_FETCH_BY_KEY && !empty($this->unique_fields)) {
|
||||||
|
if (empty($params['id'])) {
|
||||||
|
$where = array_intersect_key($params, array_flip($this->unique_fields));
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
$where = array('id' => $params['id']);
|
||||||
|
}
|
||||||
|
// Compose where clause from given field names
|
||||||
|
} else if (is_array($fetch) && !empty($fetch)) {
|
||||||
|
$where = array_intersect_key($params, array_flip($fetch));
|
||||||
|
// Use entire params array for where clause
|
||||||
} else {
|
} else {
|
||||||
self::set_properties($this, $this->optional_fields);//apply defaults for optional fields
|
$where = $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Attempt to load from database
|
||||||
|
if ($data = $this->fetch($where)) {
|
||||||
|
// Apply data from database, then data sent to constructor
|
||||||
|
self::set_properties($this, $data);
|
||||||
|
self::set_properties($this, $params);
|
||||||
|
} else {
|
||||||
|
// Apply defaults for optional fields, then data from constructor
|
||||||
|
self::set_properties($this, $this->optional_fields);
|
||||||
|
self::set_properties($this, $params);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -339,4 +389,4 @@ abstract class data_object {
|
|||||||
*/
|
*/
|
||||||
public function notify_changed($deleted) {
|
public function notify_changed($deleted) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -322,8 +322,9 @@ class completion_info {
|
|||||||
*/
|
*/
|
||||||
public function get_user_completion($user_id, $criteria) {
|
public function get_user_completion($user_id, $criteria) {
|
||||||
$params = array(
|
$params = array(
|
||||||
|
'course' => $this->course_id,
|
||||||
|
'userid' => $user_id,
|
||||||
'criteriaid' => $criteria->id,
|
'criteriaid' => $criteria->id,
|
||||||
'userid' => $user_id
|
|
||||||
);
|
);
|
||||||
|
|
||||||
$completion = new completion_criteria_completion($params);
|
$completion = new completion_criteria_completion($params);
|
||||||
@ -362,7 +363,7 @@ class completion_info {
|
|||||||
// Build array of criteria objects
|
// Build array of criteria objects
|
||||||
$this->criteria = array();
|
$this->criteria = array();
|
||||||
foreach ($records as $record) {
|
foreach ($records as $record) {
|
||||||
$this->criteria[$record->id] = completion_criteria::factory($record);
|
$this->criteria[$record->id] = completion_criteria::factory((array)$record);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user