You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2137 lines
90 KiB
2137 lines
90 KiB
<?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/>.
|
|
|
|
/**
|
|
* Workshop external API
|
|
*
|
|
* @package mod_workshop
|
|
* @category external
|
|
* @copyright 2017 Juan Leyva <juan@moodle.com>
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
* @since Moodle 3.4
|
|
*/
|
|
|
|
defined('MOODLE_INTERNAL') || die;
|
|
|
|
require_once("$CFG->libdir/externallib.php");
|
|
require_once($CFG->dirroot . '/mod/workshop/locallib.php');
|
|
|
|
use mod_workshop\external\workshop_summary_exporter;
|
|
use mod_workshop\external\submission_exporter;
|
|
use mod_workshop\external\assessment_exporter;
|
|
|
|
/**
|
|
* Workshop external functions
|
|
*
|
|
* @package mod_workshop
|
|
* @category external
|
|
* @copyright 2017 Juan Leyva <juan@moodle.com>
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
* @since Moodle 3.4
|
|
*/
|
|
class mod_workshop_external extends external_api {
|
|
|
|
/**
|
|
* Describes the parameters for get_workshops_by_courses.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_workshops_by_courses_parameters() {
|
|
return new external_function_parameters (
|
|
array(
|
|
'courseids' => new external_multiple_structure(
|
|
new external_value(PARAM_INT, 'Course id'), 'Array of course ids', VALUE_DEFAULT, array()
|
|
),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns a list of workshops in a provided list of courses.
|
|
* If no list is provided all workshops that the user can view will be returned.
|
|
*
|
|
* @param array $courseids course ids
|
|
* @return array of warnings and workshops
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_workshops_by_courses($courseids = array()) {
|
|
global $PAGE;
|
|
|
|
$warnings = array();
|
|
$returnedworkshops = array();
|
|
|
|
$params = array(
|
|
'courseids' => $courseids,
|
|
);
|
|
$params = self::validate_parameters(self::get_workshops_by_courses_parameters(), $params);
|
|
|
|
$mycourses = array();
|
|
if (empty($params['courseids'])) {
|
|
$mycourses = enrol_get_my_courses();
|
|
$params['courseids'] = array_keys($mycourses);
|
|
}
|
|
|
|
// Ensure there are courseids to loop through.
|
|
if (!empty($params['courseids'])) {
|
|
|
|
list($courses, $warnings) = external_util::validate_courses($params['courseids'], $mycourses);
|
|
$output = $PAGE->get_renderer('core');
|
|
|
|
// Get the workshops in this course, this function checks users visibility permissions.
|
|
// We can avoid then additional validate_context calls.
|
|
$workshops = get_all_instances_in_courses("workshop", $courses);
|
|
foreach ($workshops as $workshop) {
|
|
|
|
$context = context_module::instance($workshop->coursemodule);
|
|
// Remove fields that are not from the workshop (added by get_all_instances_in_courses).
|
|
unset($workshop->coursemodule, $workshop->context, $workshop->visible, $workshop->section, $workshop->groupmode,
|
|
$workshop->groupingid);
|
|
|
|
$exporter = new workshop_summary_exporter($workshop, array('context' => $context));
|
|
$returnedworkshops[] = $exporter->export($output);
|
|
}
|
|
}
|
|
|
|
$result = array(
|
|
'workshops' => $returnedworkshops,
|
|
'warnings' => $warnings
|
|
);
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* Describes the get_workshops_by_courses return value.
|
|
*
|
|
* @return external_single_structure
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_workshops_by_courses_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'workshops' => new external_multiple_structure(
|
|
workshop_summary_exporter::get_read_structure()
|
|
),
|
|
'warnings' => new external_warnings(),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Utility function for validating a workshop.
|
|
*
|
|
* @param int $workshopid workshop instance id
|
|
* @return array array containing the workshop object, course, context and course module objects
|
|
* @since Moodle 3.4
|
|
*/
|
|
protected static function validate_workshop($workshopid) {
|
|
global $DB, $USER;
|
|
|
|
// Request and permission validation.
|
|
$workshop = $DB->get_record('workshop', array('id' => $workshopid), '*', MUST_EXIST);
|
|
list($course, $cm) = get_course_and_cm_from_instance($workshop, 'workshop');
|
|
|
|
$context = context_module::instance($cm->id);
|
|
self::validate_context($context);
|
|
|
|
$workshop = new workshop($workshop, $cm, $course);
|
|
|
|
return array($workshop, $course, $cm, $context);
|
|
}
|
|
|
|
|
|
/**
|
|
* Describes the parameters for get_workshop_access_information.
|
|
*
|
|
* @return external_external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_workshop_access_information_parameters() {
|
|
return new external_function_parameters (
|
|
array(
|
|
'workshopid' => new external_value(PARAM_INT, 'Workshop instance id.')
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Return access information for a given workshop.
|
|
*
|
|
* @param int $workshopid workshop instance id
|
|
* @return array of warnings and the access information
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function get_workshop_access_information($workshopid) {
|
|
global $USER;
|
|
|
|
$params = self::validate_parameters(self::get_workshop_access_information_parameters(), array('workshopid' => $workshopid));
|
|
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($params['workshopid']);
|
|
|
|
$result = array();
|
|
// Return all the available capabilities.
|
|
$capabilities = load_capability_def('mod_workshop');
|
|
foreach ($capabilities as $capname => $capdata) {
|
|
// Get fields like cansubmit so it is consistent with the access_information function implemented in other modules.
|
|
$field = 'can' . str_replace('mod/workshop:', '', $capname);
|
|
$result[$field] = has_capability($capname, $context);
|
|
}
|
|
|
|
// Now, specific features access information.
|
|
$result['creatingsubmissionallowed'] = $workshop->creating_submission_allowed($USER->id);
|
|
$result['modifyingsubmissionallowed'] = $workshop->modifying_submission_allowed($USER->id);
|
|
$result['assessingallowed'] = $workshop->assessing_allowed($USER->id);
|
|
$result['assessingexamplesallowed'] = $workshop->assessing_examples_allowed();
|
|
if (is_null($result['assessingexamplesallowed'])) {
|
|
$result['assessingexamplesallowed'] = false;
|
|
}
|
|
$result['examplesassessedbeforesubmission'] = $workshop->check_examples_assessed_before_submission($USER->id);
|
|
list($result['examplesassessedbeforeassessment'], $code) = $workshop->check_examples_assessed_before_assessment($USER->id);
|
|
|
|
$result['warnings'] = array();
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* Describes the get_workshop_access_information return value.
|
|
*
|
|
* @return external_single_structure
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_workshop_access_information_returns() {
|
|
|
|
$structure = array(
|
|
'creatingsubmissionallowed' => new external_value(PARAM_BOOL,
|
|
'Is the given user allowed to create their submission?'),
|
|
'modifyingsubmissionallowed' => new external_value(PARAM_BOOL,
|
|
'Is the user allowed to modify his existing submission?'),
|
|
'assessingallowed' => new external_value(PARAM_BOOL,
|
|
'Is the user allowed to create/edit his assessments?'),
|
|
'assessingexamplesallowed' => new external_value(PARAM_BOOL,
|
|
'Are reviewers allowed to create/edit their assessments of the example submissions?.'),
|
|
'examplesassessedbeforesubmission' => new external_value(PARAM_BOOL,
|
|
'Whether the given user has assessed all his required examples before submission
|
|
(always true if there are not examples to assess or not configured to check before submission).'),
|
|
'examplesassessedbeforeassessment' => new external_value(PARAM_BOOL,
|
|
'Whether the given user has assessed all his required examples before assessment
|
|
(always true if there are not examples to assessor not configured to check before assessment).'),
|
|
'warnings' => new external_warnings()
|
|
);
|
|
|
|
$capabilities = load_capability_def('mod_workshop');
|
|
foreach ($capabilities as $capname => $capdata) {
|
|
// Get fields like cansubmit so it is consistent with the access_information function implemented in other modules.
|
|
$field = 'can' . str_replace('mod/workshop:', '', $capname);
|
|
$structure[$field] = new external_value(PARAM_BOOL, 'Whether the user has the capability ' . $capname . ' allowed.');
|
|
}
|
|
|
|
return new external_single_structure($structure);
|
|
}
|
|
|
|
/**
|
|
* Describes the parameters for get_user_plan.
|
|
*
|
|
* @return external_external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_user_plan_parameters() {
|
|
return new external_function_parameters (
|
|
array(
|
|
'workshopid' => new external_value(PARAM_INT, 'Workshop instance id.'),
|
|
'userid' => new external_value(PARAM_INT, 'User id (empty or 0 for current user).', VALUE_DEFAULT, 0),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Return the planner information for the given user.
|
|
*
|
|
* @param int $workshopid workshop instance id
|
|
* @param int $userid user id
|
|
* @return array of warnings and the user plan
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function get_user_plan($workshopid, $userid = 0) {
|
|
global $USER;
|
|
|
|
$params = array(
|
|
'workshopid' => $workshopid,
|
|
'userid' => $userid,
|
|
);
|
|
$params = self::validate_parameters(self::get_user_plan_parameters(), $params);
|
|
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($params['workshopid']);
|
|
|
|
// Extra checks so only users with permissions can view other users plans.
|
|
if (empty($params['userid']) || $params['userid'] == $USER->id) {
|
|
$userid = $USER->id;
|
|
} else {
|
|
require_capability('moodle/course:manageactivities', $context);
|
|
$user = core_user::get_user($params['userid'], '*', MUST_EXIST);
|
|
core_user::require_active_user($user);
|
|
if (!$workshop->check_group_membership($user->id)) {
|
|
throw new moodle_exception('notingroup');
|
|
}
|
|
$userid = $user->id;
|
|
}
|
|
|
|
// Get the user plan information ready for external functions.
|
|
$userplan = new workshop_user_plan($workshop, $userid);
|
|
$userplan = array('phases' => $userplan->phases, 'examples' => $userplan->get_examples());
|
|
foreach ($userplan['phases'] as $phasecode => $phase) {
|
|
$phase->code = $phasecode;
|
|
$userplan['phases'][$phasecode] = (array) $phase;
|
|
foreach ($userplan['phases'][$phasecode]['tasks'] as $taskcode => $task) {
|
|
$task->code = $taskcode;
|
|
if ($task->link instanceof moodle_url) {
|
|
$task->link = $task->link->out(false);
|
|
}
|
|
$userplan['phases'][$phasecode]['tasks'][$taskcode] = (array) $task;
|
|
}
|
|
foreach ($userplan['phases'][$phasecode]['actions'] as $actioncode => $action) {
|
|
if ($action->url instanceof moodle_url) {
|
|
$action->url = $action->url->out(false);
|
|
}
|
|
$userplan['phases'][$phasecode]['actions'][$actioncode] = (array) $action;
|
|
}
|
|
}
|
|
|
|
$result['userplan'] = $userplan;
|
|
$result['warnings'] = array();
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* Describes the get_user_plan return value.
|
|
*
|
|
* @return external_single_structure
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_user_plan_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'userplan' => new external_single_structure(
|
|
array(
|
|
'phases' => new external_multiple_structure(
|
|
new external_single_structure(
|
|
array(
|
|
'code' => new external_value(PARAM_INT, 'Phase code.'),
|
|
'title' => new external_value(PARAM_NOTAGS, 'Phase title.'),
|
|
'active' => new external_value(PARAM_BOOL, 'Whether is the active task.'),
|
|
'tasks' => new external_multiple_structure(
|
|
new external_single_structure(
|
|
array(
|
|
'code' => new external_value(PARAM_ALPHA, 'Task code.'),
|
|
'title' => new external_value(PARAM_RAW, 'Task title.'),
|
|
'link' => new external_value(PARAM_URL, 'Link to task.'),
|
|
'details' => new external_value(PARAM_RAW, 'Task details.', VALUE_OPTIONAL),
|
|
'completed' => new external_value(PARAM_NOTAGS,
|
|
'Completion information (maybe empty, maybe a boolean or generic info.'),
|
|
)
|
|
)
|
|
),
|
|
'actions' => new external_multiple_structure(
|
|
new external_single_structure(
|
|
array(
|
|
'type' => new external_value(PARAM_ALPHA, 'Action type.', VALUE_OPTIONAL),
|
|
'label' => new external_value(PARAM_RAW, 'Action label.', VALUE_OPTIONAL),
|
|
'url' => new external_value(PARAM_URL, 'Link to action.'),
|
|
'method' => new external_value(PARAM_ALPHA, 'Get or post.', VALUE_OPTIONAL),
|
|
)
|
|
)
|
|
),
|
|
)
|
|
)
|
|
),
|
|
'examples' => new external_multiple_structure(
|
|
new external_single_structure(
|
|
array(
|
|
'id' => new external_value(PARAM_INT, 'Example submission id.'),
|
|
'title' => new external_value(PARAM_RAW, 'Example submission title.'),
|
|
'assessmentid' => new external_value(PARAM_INT, 'Example submission assessment id.'),
|
|
'grade' => new external_value(PARAM_FLOAT, 'The submission grade.'),
|
|
'gradinggrade' => new external_value(PARAM_FLOAT, 'The assessment grade.'),
|
|
)
|
|
)
|
|
),
|
|
)
|
|
),
|
|
'warnings' => new external_warnings(),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Describes the parameters for view_workshop.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function view_workshop_parameters() {
|
|
return new external_function_parameters (
|
|
array(
|
|
'workshopid' => new external_value(PARAM_INT, 'Workshop instance id'),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Trigger the course module viewed event and update the module completion status.
|
|
*
|
|
* @param int $workshopid workshop instance id
|
|
* @return array of warnings and status result
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function view_workshop($workshopid) {
|
|
|
|
$params = array('workshopid' => $workshopid);
|
|
$params = self::validate_parameters(self::view_workshop_parameters(), $params);
|
|
$warnings = array();
|
|
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($params['workshopid']);
|
|
|
|
$workshop->set_module_viewed();
|
|
|
|
$result = array(
|
|
'status' => true,
|
|
'warnings' => $warnings,
|
|
);
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* Describes the view_workshop return value.
|
|
*
|
|
* @return external_single_structure
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function view_workshop_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'status' => new external_value(PARAM_BOOL, 'status: true if success'),
|
|
'warnings' => new external_warnings(),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function add_submission_parameters() {
|
|
return new external_function_parameters(array(
|
|
'workshopid' => new external_value(PARAM_INT, 'Workshop id'),
|
|
'title' => new external_value(PARAM_TEXT, 'Submission title'),
|
|
'content' => new external_value(PARAM_RAW, 'Submission text content', VALUE_DEFAULT, ''),
|
|
'contentformat' => new external_value(PARAM_INT, 'The format used for the content', VALUE_DEFAULT, FORMAT_MOODLE),
|
|
'inlineattachmentsid' => new external_value(PARAM_INT, 'The draft file area id for inline attachments in the content',
|
|
VALUE_DEFAULT, 0),
|
|
'attachmentsid' => new external_value(PARAM_INT, 'The draft file area id for attachments', VALUE_DEFAULT, 0),
|
|
));
|
|
}
|
|
|
|
/**
|
|
* Add a new submission to a given workshop.
|
|
*
|
|
* @param int $workshopid the workshop id
|
|
* @param string $title the submission title
|
|
* @param string $content the submission text content
|
|
* @param int $contentformat the format used for the content
|
|
* @param int $inlineattachmentsid the draft file area id for inline attachments in the content
|
|
* @param int $attachmentsid the draft file area id for attachments
|
|
* @return array Containing the new created submission id and warnings.
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function add_submission($workshopid, $title, $content = '', $contentformat = FORMAT_MOODLE,
|
|
$inlineattachmentsid = 0, $attachmentsid = 0) {
|
|
global $USER;
|
|
|
|
$params = self::validate_parameters(self::add_submission_parameters(), array(
|
|
'workshopid' => $workshopid,
|
|
'title' => $title,
|
|
'content' => $content,
|
|
'contentformat' => $contentformat,
|
|
'inlineattachmentsid' => $inlineattachmentsid,
|
|
'attachmentsid' => $attachmentsid,
|
|
));
|
|
$warnings = array();
|
|
|
|
// Get and validate the workshop.
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($params['workshopid']);
|
|
require_capability('mod/workshop:submit', $context);
|
|
|
|
// Check if we can submit now.
|
|
$canaddsubmission = $workshop->creating_submission_allowed($USER->id);
|
|
$canaddsubmission = $canaddsubmission && $workshop->check_examples_assessed_before_submission($USER->id);
|
|
if (!$canaddsubmission) {
|
|
throw new moodle_exception('nopermissions', 'error', '', 'add submission');
|
|
}
|
|
|
|
// Prepare the submission object.
|
|
$submission = new stdClass;
|
|
$submission->id = null;
|
|
$submission->cmid = $cm->id;
|
|
$submission->example = 0;
|
|
$submission->title = trim($params['title']);
|
|
$submission->content_editor = array(
|
|
'text' => $params['content'],
|
|
'format' => $params['contentformat'],
|
|
'itemid' => $params['inlineattachmentsid'],
|
|
);
|
|
$submission->attachment_filemanager = $params['attachmentsid'];
|
|
|
|
if (empty($submission->title)) {
|
|
throw new moodle_exception('errorinvalidparam', 'webservice', '', 'title');
|
|
}
|
|
|
|
$errors = $workshop->validate_submission_data((array) $submission);
|
|
// We can get several errors, return them in warnings.
|
|
if (!empty($errors)) {
|
|
$submission->id = 0;
|
|
foreach ($errors as $itemname => $message) {
|
|
$warnings[] = array(
|
|
'item' => $itemname,
|
|
'itemid' => 0,
|
|
'warningcode' => 'fielderror',
|
|
'message' => s($message)
|
|
);
|
|
}
|
|
return array(
|
|
'status' => false,
|
|
'warnings' => $warnings
|
|
);
|
|
} else {
|
|
$submission->id = $workshop->edit_submission($submission);
|
|
return array(
|
|
'status' => true,
|
|
'submissionid' => $submission->id,
|
|
'warnings' => $warnings
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function return value.
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function add_submission_returns() {
|
|
return new external_single_structure(array(
|
|
'status' => new external_value(PARAM_BOOL, 'True if the submission was created false otherwise.'),
|
|
'submissionid' => new external_value(PARAM_INT, 'New workshop submission id.', VALUE_OPTIONAL),
|
|
'warnings' => new external_warnings()
|
|
));
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function update_submission_parameters() {
|
|
return new external_function_parameters(array(
|
|
'submissionid' => new external_value(PARAM_INT, 'Submission id'),
|
|
'title' => new external_value(PARAM_TEXT, 'Submission title'),
|
|
'content' => new external_value(PARAM_RAW, 'Submission text content', VALUE_DEFAULT, ''),
|
|
'contentformat' => new external_value(PARAM_INT, 'The format used for the content', VALUE_DEFAULT, FORMAT_MOODLE),
|
|
'inlineattachmentsid' => new external_value(PARAM_INT, 'The draft file area id for inline attachments in the content',
|
|
VALUE_DEFAULT, 0),
|
|
'attachmentsid' => new external_value(PARAM_INT, 'The draft file area id for attachments', VALUE_DEFAULT, 0),
|
|
));
|
|
}
|
|
|
|
|
|
/**
|
|
* Updates the given submission.
|
|
*
|
|
* @param int $submissionid the submission id
|
|
* @param string $title the submission title
|
|
* @param string $content the submission text content
|
|
* @param int $contentformat the format used for the content
|
|
* @param int $inlineattachmentsid the draft file area id for inline attachments in the content
|
|
* @param int $attachmentsid the draft file area id for attachments
|
|
* @return array whether the submission was updated and warnings.
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function update_submission($submissionid, $title, $content = '', $contentformat = FORMAT_MOODLE,
|
|
$inlineattachmentsid = 0, $attachmentsid = 0) {
|
|
global $USER, $DB;
|
|
|
|
$params = self::validate_parameters(self::update_submission_parameters(), array(
|
|
'submissionid' => $submissionid,
|
|
'title' => $title,
|
|
'content' => $content,
|
|
'contentformat' => $contentformat,
|
|
'inlineattachmentsid' => $inlineattachmentsid,
|
|
'attachmentsid' => $attachmentsid,
|
|
));
|
|
$warnings = array();
|
|
|
|
// Get and validate the submission and workshop.
|
|
$submission = $DB->get_record('workshop_submissions', array('id' => $params['submissionid']), '*', MUST_EXIST);
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($submission->workshopid);
|
|
require_capability('mod/workshop:submit', $context);
|
|
|
|
// Check if we can update the submission.
|
|
$canupdatesubmission = $submission->authorid == $USER->id;
|
|
$canupdatesubmission = $canupdatesubmission && $workshop->modifying_submission_allowed($USER->id);
|
|
$canupdatesubmission = $canupdatesubmission && $workshop->check_examples_assessed_before_submission($USER->id);
|
|
if (!$canupdatesubmission) {
|
|
throw new moodle_exception('nopermissions', 'error', '', 'update submission');
|
|
}
|
|
|
|
// Prepare the submission object.
|
|
$submission->title = trim($params['title']);
|
|
if (empty($submission->title)) {
|
|
throw new moodle_exception('errorinvalidparam', 'webservice', '', 'title');
|
|
}
|
|
$submission->content_editor = array(
|
|
'text' => $params['content'],
|
|
'format' => $params['contentformat'],
|
|
'itemid' => $params['inlineattachmentsid'],
|
|
);
|
|
$submission->attachment_filemanager = $params['attachmentsid'];
|
|
|
|
$errors = $workshop->validate_submission_data((array) $submission);
|
|
// We can get several errors, return them in warnings.
|
|
if (!empty($errors)) {
|
|
$status = false;
|
|
foreach ($errors as $itemname => $message) {
|
|
$warnings[] = array(
|
|
'item' => $itemname,
|
|
'itemid' => 0,
|
|
'warningcode' => 'fielderror',
|
|
'message' => s($message)
|
|
);
|
|
}
|
|
} else {
|
|
$status = true;
|
|
$submission->id = $workshop->edit_submission($submission);
|
|
}
|
|
|
|
return array(
|
|
'status' => $status,
|
|
'warnings' => $warnings
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function return value.
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function update_submission_returns() {
|
|
return new external_single_structure(array(
|
|
'status' => new external_value(PARAM_BOOL, 'True if the submission was updated false otherwise.'),
|
|
'warnings' => new external_warnings()
|
|
));
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function delete_submission_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'submissionid' => new external_value(PARAM_INT, 'Submission id'),
|
|
)
|
|
);
|
|
}
|
|
|
|
|
|
/**
|
|
* Deletes the given submission.
|
|
*
|
|
* @param int $submissionid the submission id.
|
|
* @return array containing the result status and warnings.
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function delete_submission($submissionid) {
|
|
global $USER, $DB;
|
|
|
|
$params = self::validate_parameters(self::delete_submission_parameters(), array('submissionid' => $submissionid));
|
|
$warnings = array();
|
|
|
|
// Get and validate the submission and workshop.
|
|
$submission = $DB->get_record('workshop_submissions', array('id' => $params['submissionid']), '*', MUST_EXIST);
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($submission->workshopid);
|
|
|
|
// Check if we can delete the submission.
|
|
if (!has_capability('mod/workshop:deletesubmissions', $context)) {
|
|
require_capability('mod/workshop:submit', $context);
|
|
// We can delete our own submission, on time and not yet assessed.
|
|
$candeletesubmission = $submission->authorid == $USER->id;
|
|
$candeletesubmission = $candeletesubmission && $workshop->modifying_submission_allowed($USER->id);
|
|
$candeletesubmission = $candeletesubmission && count($workshop->get_assessments_of_submission($submission->id)) == 0;
|
|
if (!$candeletesubmission) {
|
|
throw new moodle_exception('nopermissions', 'error', '', 'delete submission');
|
|
}
|
|
}
|
|
|
|
$workshop->delete_submission($submission);
|
|
|
|
return array(
|
|
'status' => true,
|
|
'warnings' => $warnings
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function return value.
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function delete_submission_returns() {
|
|
return new external_single_structure(array(
|
|
'status' => new external_value(PARAM_BOOL, 'True if the submission was deleted.'),
|
|
'warnings' => new external_warnings()
|
|
));
|
|
}
|
|
|
|
/**
|
|
* Helper method for returning the submission data according the current user capabilities and current phase.
|
|
*
|
|
* @param stdClass $submission the submission data
|
|
* @param workshop $workshop the workshop class
|
|
* @param bool $canviewauthorpublished whether the user has the capability mod/workshop:viewauthorpublished on
|
|
* @param bool $canviewauthornames whether the user has the capability mod/workshop:vviewauthornames on
|
|
* @param bool $canviewallsubmissions whether the user has the capability mod/workshop:viewallsubmissions on
|
|
* @return stdClass object with the submission data filtered
|
|
* @since Moodle 3.4
|
|
*/
|
|
protected static function prepare_submission_for_external($submission, workshop $workshop, $canviewauthorpublished = null,
|
|
$canviewauthornames = null, $canviewallsubmissions = null) {
|
|
global $USER;
|
|
|
|
if (is_null($canviewauthorpublished)) {
|
|
$canviewauthorpublished = has_capability('mod/workshop:viewauthorpublished', $workshop->context);
|
|
}
|
|
if (is_null($canviewauthornames)) {
|
|
$canviewauthornames = has_capability('mod/workshop:viewauthornames', $workshop->context);
|
|
}
|
|
if (is_null($canviewallsubmissions)) {
|
|
$canviewallsubmissions = has_capability('mod/workshop:viewallsubmissions', $workshop->context);
|
|
}
|
|
|
|
$ownsubmission = $submission->authorid == $USER->id;
|
|
if (!$canviewauthornames && !$ownsubmission) {
|
|
$submission->authorid = 0;
|
|
}
|
|
|
|
// Remove grade, gradeover, gradeoverby, feedbackauthor and timegraded for non-teachers or invalid phase.
|
|
// WS mod_workshop_external::get_grades should be used for retrieving grades by students.
|
|
if ($workshop->phase < workshop::PHASE_EVALUATION || !$canviewallsubmissions) {
|
|
$properties = submission_exporter::properties_definition();
|
|
foreach ($properties as $attribute => $settings) {
|
|
// Special case, the feedbackauthor (and who did it) should be returned if the workshop is closed and
|
|
// the user can view it.
|
|
if (($attribute == 'feedbackauthor' || $attribute == 'gradeoverby') &&
|
|
$workshop->phase == workshop::PHASE_CLOSED && $ownsubmission) {
|
|
continue;
|
|
}
|
|
if (!empty($settings['optional'])) {
|
|
unset($submission->{$attribute});
|
|
}
|
|
}
|
|
}
|
|
return $submission;
|
|
}
|
|
|
|
/**
|
|
* Returns description of method parameters
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_submissions_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'workshopid' => new external_value(PARAM_INT, 'Workshop instance id.'),
|
|
'userid' => new external_value(PARAM_INT, 'Get submissions done by this user. Use 0 or empty for the current user',
|
|
VALUE_DEFAULT, 0),
|
|
'groupid' => new external_value(PARAM_INT, 'Group id, 0 means that the function will determine the user group.
|
|
It will return submissions done by users in the given group.',
|
|
VALUE_DEFAULT, 0),
|
|
'page' => new external_value(PARAM_INT, 'The page of records to return.', VALUE_DEFAULT, 0),
|
|
'perpage' => new external_value(PARAM_INT, 'The number of records to return per page.', VALUE_DEFAULT, 0),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Retrieves all the workshop submissions visible by the current user or the one done by the given user
|
|
* (except example submissions).
|
|
*
|
|
* @param int $workshopid the workshop instance id
|
|
* @param int $userid get submissions done by this user
|
|
* @param int $groupid (optional) group id, 0 means that the function will determine the user group
|
|
* @param int $page page of records to return
|
|
* @param int $perpage number of records to return per page
|
|
* @return array of warnings and the entries
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function get_submissions($workshopid, $userid = 0, $groupid = 0, $page = 0, $perpage = 0) {
|
|
global $PAGE, $USER;
|
|
|
|
$params = array('workshopid' => $workshopid, 'userid' => $userid, 'groupid' => $groupid,
|
|
'page' => $page, 'perpage' => $perpage);
|
|
$params = self::validate_parameters(self::get_submissions_parameters(), $params);
|
|
$submissions = $warnings = array();
|
|
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($params['workshopid']);
|
|
|
|
if (empty($params['groupid'])) {
|
|
// Check to see if groups are being used here.
|
|
if ($groupmode = groups_get_activity_groupmode($cm)) {
|
|
$groupid = groups_get_activity_group($cm);
|
|
// Determine is the group is visible to user (this is particullary for the group 0 -> all groups).
|
|
if (!groups_group_visible($groupid, $course, $cm)) {
|
|
throw new moodle_exception('notingroup');
|
|
}
|
|
} else {
|
|
$groupid = 0;
|
|
}
|
|
}
|
|
|
|
if (!empty($params['userid']) && $params['userid'] != $USER->id) {
|
|
$user = core_user::get_user($params['userid'], '*', MUST_EXIST);
|
|
core_user::require_active_user($user);
|
|
if (!$workshop->check_group_membership($user->id)) {
|
|
throw new moodle_exception('notingroup');
|
|
}
|
|
}
|
|
|
|
$totalfilesize = 0;
|
|
list($submissionsrecords, $totalcount) =
|
|
$workshop->get_visible_submissions($params['userid'], $groupid, $params['page'], $params['perpage']);
|
|
|
|
if ($totalcount) {
|
|
|
|
$canviewauthorpublished = has_capability('mod/workshop:viewauthorpublished', $context);
|
|
$canviewauthornames = has_capability('mod/workshop:viewauthornames', $context);
|
|
$canviewallsubmissions = has_capability('mod/workshop:viewallsubmissions', $context);
|
|
|
|
$related = array('context' => $context);
|
|
foreach ($submissionsrecords as $submission) {
|
|
$submission = self::prepare_submission_for_external($submission, $workshop, $canviewauthorpublished,
|
|
$canviewauthornames, $canviewallsubmissions);
|
|
|
|
$exporter = new submission_exporter($submission, $related);
|
|
$submissions[] = $exporter->export($PAGE->get_renderer('core'));
|
|
}
|
|
|
|
// Retrieve total files size for the submissions (so external clients know how many data they'd need to download).
|
|
$fs = get_file_storage();
|
|
$files = $fs->get_area_files($context->id, 'mod_workshop', array('submission_content', 'submission_attachment'));
|
|
foreach ($files as $file) {
|
|
if ($file->is_directory()) {
|
|
continue;
|
|
}
|
|
$totalfilesize += $file->get_filesize();
|
|
}
|
|
}
|
|
|
|
return array(
|
|
'submissions' => $submissions,
|
|
'totalcount' => $totalcount,
|
|
'totalfilesize' => $totalfilesize,
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns description of method result value
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_submissions_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'submissions' => new external_multiple_structure(
|
|
submission_exporter::get_read_structure()
|
|
),
|
|
'totalcount' => new external_value(PARAM_INT, 'Total count of submissions.'),
|
|
'totalfilesize' => new external_value(PARAM_INT, 'Total size (bytes) of the files attached to all the
|
|
submissions (even the ones not returned due to pagination).'),
|
|
'warnings' => new external_warnings()
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Helper method for validating a submission.
|
|
*
|
|
* @param stdClass $submission submission object
|
|
* @param workshop $workshop workshop instance
|
|
* @return void
|
|
* @since Moodle 3.4
|
|
*/
|
|
protected static function validate_submission($submission, workshop $workshop) {
|
|
global $USER;
|
|
|
|
$workshopclosed = $workshop->phase == workshop::PHASE_CLOSED;
|
|
$canviewpublished = has_capability('mod/workshop:viewpublishedsubmissions', $workshop->context);
|
|
|
|
$canview = $submission->authorid == $USER->id; // I did it.
|
|
$canview = $canview || !empty($workshop->get_assessment_of_submission_by_user($submission->id, $USER->id)); // I reviewed.
|
|
$canview = $canview || has_capability('mod/workshop:viewallsubmissions', $workshop->context); // I can view all.
|
|
$canview = $canview || ($submission->published && $workshopclosed && $canviewpublished); // It has been published.
|
|
|
|
if ($canview) {
|
|
// Here we should check if the user share group.
|
|
if ($submission->authorid != $USER->id &&
|
|
!groups_user_groups_visible($workshop->course, $submission->authorid, $workshop->cm)) {
|
|
throw new moodle_exception('notingroup');
|
|
}
|
|
} else {
|
|
throw new moodle_exception('nopermissions', 'error', '', 'view submission');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_submission_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'submissionid' => new external_value(PARAM_INT, 'Submission id'),
|
|
)
|
|
);
|
|
}
|
|
|
|
|
|
/**
|
|
* Retrieves the given submission.
|
|
*
|
|
* @param int $submissionid the submission id
|
|
* @return array containing the submission and warnings.
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function get_submission($submissionid) {
|
|
global $USER, $DB, $PAGE;
|
|
|
|
$params = self::validate_parameters(self::get_submission_parameters(), array('submissionid' => $submissionid));
|
|
$warnings = array();
|
|
|
|
// Get and validate the submission and workshop.
|
|
$submission = $DB->get_record('workshop_submissions', array('id' => $params['submissionid']), '*', MUST_EXIST);
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($submission->workshopid);
|
|
|
|
self::validate_submission($submission, $workshop);
|
|
|
|
$submission = self::prepare_submission_for_external($submission, $workshop);
|
|
|
|
$related = array('context' => $context);
|
|
$exporter = new submission_exporter($submission, $related);
|
|
return array(
|
|
'submission' => $exporter->export($PAGE->get_renderer('core')),
|
|
'warnings' => $warnings
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns description of method result value
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_submission_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'submission' => submission_exporter::get_read_structure(),
|
|
'warnings' => new external_warnings()
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Helper method for validating if the current user can view the submission assessments.
|
|
*
|
|
* @param stdClass $submission submission object
|
|
* @param workshop $workshop workshop instance
|
|
* @return void
|
|
* @since Moodle 3.4
|
|
*/
|
|
protected static function check_view_submission_assessments($submission, workshop $workshop) {
|
|
global $USER;
|
|
|
|
$ownsubmission = $submission->authorid == $USER->id;
|
|
$canview = has_capability('mod/workshop:viewallassessments', $workshop->context) ||
|
|
($ownsubmission && $workshop->assessments_available());
|
|
|
|
if ($canview) {
|
|
// Here we should check if the user share group.
|
|
if ($submission->authorid != $USER->id &&
|
|
!groups_user_groups_visible($workshop->course, $submission->authorid, $workshop->cm)) {
|
|
throw new moodle_exception('notingroup');
|
|
}
|
|
} else {
|
|
throw new moodle_exception('nopermissions', 'error', '', 'view assessment');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Helper method for returning the assessment data according the current user capabilities and current phase.
|
|
*
|
|
* @param stdClass $assessment the assessment data
|
|
* @param workshop $workshop the workshop class
|
|
* @return stdClass object with the assessment data filtered or null if is not viewable yet
|
|
* @since Moodle 3.4
|
|
*/
|
|
protected static function prepare_assessment_for_external($assessment, workshop $workshop) {
|
|
global $USER;
|
|
static $canviewallassessments = null;
|
|
static $canviewreviewers = null;
|
|
static $canoverridegrades = null;
|
|
|
|
// Remove all the properties that does not belong to the assessment table.
|
|
$properties = assessment_exporter::properties_definition();
|
|
foreach ($assessment as $key => $value) {
|
|
if (!isset($properties[$key])) {
|
|
unset($assessment->{$key});
|
|
}
|
|
}
|
|
|
|
if (is_null($canviewallassessments)) {
|
|
$canviewallassessments = has_capability('mod/workshop:viewallassessments', $workshop->context);
|
|
}
|
|
if (is_null($canviewreviewers)) {
|
|
$canviewreviewers = has_capability('mod/workshop:viewreviewernames', $workshop->context);
|
|
}
|
|
if (is_null($canoverridegrades)) {
|
|
$canoverridegrades = has_capability('mod/workshop:overridegrades', $workshop->context);
|
|
}
|
|
|
|
$isreviewer = $assessment->reviewerid == $USER->id;
|
|
|
|
if (!$isreviewer && is_null($assessment->grade) && !$canviewallassessments) {
|
|
// Students do not see peer-assessment that are not graded yet.
|
|
return null;
|
|
}
|
|
|
|
// Remove the feedback for the reviewer if:
|
|
// I can't see it in the evaluation phase because I'm not a teacher or the reviewer AND
|
|
// I can't see it in the assessment phase because I'm not a teacher.
|
|
if (($workshop->phase < workshop::PHASE_EVALUATION || !($isreviewer || $canviewallassessments)) &&
|
|
($workshop->phase < workshop::PHASE_ASSESSMENT || !$canviewallassessments) ) {
|
|
// Remove all the feedback information (all the optional fields).
|
|
foreach ($properties as $attribute => $settings) {
|
|
if (!empty($settings['optional'])) {
|
|
unset($assessment->{$attribute});
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!$isreviewer && !$canviewreviewers) {
|
|
$assessment->reviewerid = 0;
|
|
}
|
|
|
|
return $assessment;
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_submission_assessments_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'submissionid' => new external_value(PARAM_INT, 'Submission id'),
|
|
)
|
|
);
|
|
}
|
|
|
|
|
|
/**
|
|
* Retrieves the given submission assessments.
|
|
*
|
|
* @param int $submissionid the submission id
|
|
* @return array containing the assessments and warnings.
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function get_submission_assessments($submissionid) {
|
|
global $USER, $DB, $PAGE;
|
|
|
|
$params = self::validate_parameters(self::get_submission_assessments_parameters(), array('submissionid' => $submissionid));
|
|
$warnings = $assessments = array();
|
|
|
|
// Get and validate the submission and workshop.
|
|
$submission = $DB->get_record('workshop_submissions', array('id' => $params['submissionid']), '*', MUST_EXIST);
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($submission->workshopid);
|
|
|
|
// Check that we can get the assessments and get them.
|
|
self::check_view_submission_assessments($submission, $workshop);
|
|
$assessmentsrecords = $workshop->get_assessments_of_submission($submission->id);
|
|
|
|
$related = array('context' => $context);
|
|
foreach ($assessmentsrecords as $assessment) {
|
|
$assessment = self::prepare_assessment_for_external($assessment, $workshop);
|
|
if (empty($assessment)) {
|
|
continue;
|
|
}
|
|
$exporter = new assessment_exporter($assessment, $related);
|
|
$assessments[] = $exporter->export($PAGE->get_renderer('core'));
|
|
}
|
|
|
|
return array(
|
|
'assessments' => $assessments,
|
|
'warnings' => $warnings
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns description of method result value
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_submission_assessments_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'assessments' => new external_multiple_structure(
|
|
assessment_exporter::get_read_structure()
|
|
),
|
|
'warnings' => new external_warnings()
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_assessment_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'assessmentid' => new external_value(PARAM_INT, 'Assessment id'),
|
|
)
|
|
);
|
|
}
|
|
|
|
|
|
/**
|
|
* Retrieves the given assessment.
|
|
*
|
|
* @param int $assessmentid the assessment id
|
|
* @return array containing the assessment and warnings.
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function get_assessment($assessmentid) {
|
|
global $DB, $PAGE;
|
|
|
|
$params = self::validate_parameters(self::get_assessment_parameters(), array('assessmentid' => $assessmentid));
|
|
$warnings = array();
|
|
|
|
// Get and validate the assessment, submission and workshop.
|
|
$assessment = $DB->get_record('workshop_assessments', array('id' => $params['assessmentid']), '*', MUST_EXIST);
|
|
$submission = $DB->get_record('workshop_submissions', array('id' => $assessment->submissionid), '*', MUST_EXIST);
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($submission->workshopid);
|
|
|
|
// Check that we can get the assessment.
|
|
$workshop->check_view_assessment($assessment, $submission);
|
|
|
|
$assessment = $workshop->get_assessment_by_id($assessment->id);
|
|
$assessment = self::prepare_assessment_for_external($assessment, $workshop);
|
|
if (empty($assessment)) {
|
|
throw new moodle_exception('nopermissions', 'error', '', 'view assessment');
|
|
}
|
|
$related = array('context' => $context);
|
|
$exporter = new assessment_exporter($assessment, $related);
|
|
|
|
return array(
|
|
'assessment' => $exporter->export($PAGE->get_renderer('core')),
|
|
'warnings' => $warnings
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns description of method result value
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_assessment_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'assessment' => assessment_exporter::get_read_structure(),
|
|
'warnings' => new external_warnings()
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_assessment_form_definition_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'assessmentid' => new external_value(PARAM_INT, 'Assessment id'),
|
|
'mode' => new external_value(PARAM_ALPHA, 'The form mode (assessment or preview)', VALUE_DEFAULT, 'assessment'),
|
|
)
|
|
);
|
|
}
|
|
|
|
|
|
/**
|
|
* Retrieves the assessment form definition (data required to be able to display the assessment form).
|
|
*
|
|
* @param int $assessmentid the assessment id
|
|
* @param string $mode the form mode (assessment or preview)
|
|
* @return array containing the assessment and warnings.
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function get_assessment_form_definition($assessmentid, $mode = 'assessment') {
|
|
global $DB, $USER;
|
|
|
|
$params = self::validate_parameters(
|
|
self::get_assessment_form_definition_parameters(), array('assessmentid' => $assessmentid, 'mode' => $mode)
|
|
);
|
|
$warnings = $pending = array();
|
|
|
|
if ($params['mode'] != 'assessment' && $params['mode'] != 'preview') {
|
|
throw new invalid_parameter_exception('Invalid value for mode parameter (value: ' . $params['mode'] . ')');
|
|
}
|
|
|
|
// Get and validate the assessment, submission and workshop.
|
|
$assessment = $DB->get_record('workshop_assessments', array('id' => $params['assessmentid']), '*', MUST_EXIST);
|
|
$submission = $DB->get_record('workshop_submissions', array('id' => $assessment->submissionid), '*', MUST_EXIST);
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($submission->workshopid);
|
|
|
|
// Check we can view the assessment (so we can get the form data).
|
|
$workshop->check_view_assessment($assessment, $submission);
|
|
|
|
$cansetassessmentweight = has_capability('mod/workshop:allocate', $context);
|
|
$pending = $workshop->get_pending_assessments_by_reviewer($assessment->reviewerid, $assessment->id);
|
|
|
|
// Retrieve the data from the strategy plugin.
|
|
$strategy = $workshop->grading_strategy_instance();
|
|
$strategyname = str_replace('_strategy', '', get_class($strategy)); // Get strategy name.
|
|
$mform = $strategy->get_assessment_form(null, $params['mode'], $assessment, true,
|
|
array('editableweight' => $cansetassessmentweight, 'pending' => !empty($pending)));
|
|
$formdata = $mform->get_customdata();
|
|
|
|
$result = array(
|
|
'dimenssionscount' => $formdata['nodims'],
|
|
'descriptionfiles' => external_util::get_area_files($context->id, $strategyname, 'description'),
|
|
'warnings' => $warnings
|
|
);
|
|
// Include missing dimension fields.
|
|
for ($i = 0; $i < $formdata['nodims']; $i++) {
|
|
$formdata['fields']->{'gradeid__idx_' . $i} = 0;
|
|
$formdata['fields']->{'peercomment__idx_' . $i} = '';
|
|
}
|
|
|
|
// Convert all the form data for external.
|
|
foreach (array('options', 'fields', 'current') as $typeofdata) {
|
|
$result[$typeofdata] = array();
|
|
|
|
if (!empty($formdata[$typeofdata])) {
|
|
$alldata = (array) $formdata[$typeofdata];
|
|
foreach ($alldata as $key => $val) {
|
|
if (strpos($key, 'peercomment__idx_') === 0) {
|
|
// Format reviewer comment.
|
|
list($val, $format) = external_format_text($val, FORMAT_MOODLE, $context->id);
|
|
} else if (strpos($key, 'description__idx_')) {
|
|
// Format dimension description.
|
|
$id = str_replace('description__idx_', '', $key);
|
|
list($val, $format) = external_format_text($val, $alldata['dimensionid__idx_' . $id . 'format'],
|
|
$context->id, $strategyname, 'description', $alldata['dimensionid__idx_' . $id]);
|
|
}
|
|
$result[$typeofdata][] = array(
|
|
'name' => $key,
|
|
'value' => $val
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Get dimensions info.
|
|
$grader = $workshop->grading_strategy_instance();
|
|
$result['dimensionsinfo'] = $grader->get_dimensions_info();
|
|
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* Returns description of method result value
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_assessment_form_definition_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'dimenssionscount' => new external_value(PARAM_INT, 'The number of dimenssions used by the form.'),
|
|
'descriptionfiles' => new external_files('Files in the description text'),
|
|
'options' => new external_multiple_structure(
|
|
new external_single_structure(
|
|
array(
|
|
'name' => new external_value(PARAM_ALPHANUMEXT, 'Option name.'),
|
|
'value' => new external_value(PARAM_NOTAGS, 'Option value.')
|
|
)
|
|
), 'The form options.'
|
|
),
|
|
'fields' => new external_multiple_structure(
|
|
new external_single_structure(
|
|
array(
|
|
'name' => new external_value(PARAM_ALPHANUMEXT, 'Field name.'),
|
|
'value' => new external_value(PARAM_RAW, 'Field default value.')
|
|
)
|
|
), 'The form fields.'
|
|
),
|
|
'current' => new external_multiple_structure(
|
|
new external_single_structure(
|
|
array(
|
|
'name' => new external_value(PARAM_ALPHANUMEXT, 'Field name.'),
|
|
'value' => new external_value(PARAM_RAW, 'Current field value.')
|
|
)
|
|
), 'The current field values.'
|
|
),
|
|
'dimensionsinfo' => new external_multiple_structure(
|
|
new external_single_structure(
|
|
array(
|
|
'id' => new external_value(PARAM_INT, 'Dimension id.'),
|
|
'min' => new external_value(PARAM_FLOAT, 'Minimum grade for the dimension.'),
|
|
'max' => new external_value(PARAM_FLOAT, 'Maximum grade for the dimension.'),
|
|
'weight' => new external_value(PARAM_TEXT, 'The weight of the dimension.'),
|
|
'scale' => new external_value(PARAM_TEXT, 'Scale items (if used).', VALUE_OPTIONAL),
|
|
)
|
|
), 'The dimensions general information.'
|
|
),
|
|
'warnings' => new external_warnings()
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_reviewer_assessments_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'workshopid' => new external_value(PARAM_INT, 'Workshop instance id.'),
|
|
'userid' => new external_value(PARAM_INT, 'User id who did the assessment review (empty or 0 for current user).',
|
|
VALUE_DEFAULT, 0),
|
|
)
|
|
);
|
|
}
|
|
|
|
|
|
/**
|
|
* Retrieves all the assessments reviewed by the given user.
|
|
*
|
|
* @param int $workshopid the workshop instance id
|
|
* @param int $userid the reviewer user id
|
|
* @return array containing the user assessments and warnings.
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function get_reviewer_assessments($workshopid, $userid = 0) {
|
|
global $USER, $DB, $PAGE;
|
|
|
|
$params = self::validate_parameters(
|
|
self::get_reviewer_assessments_parameters(), array('workshopid' => $workshopid, 'userid' => $userid)
|
|
);
|
|
$warnings = $assessments = array();
|
|
|
|
// Get and validate the submission and workshop.
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($params['workshopid']);
|
|
|
|
// Extra checks so only users with permissions can view other users assessments.
|
|
if (empty($params['userid']) || $params['userid'] == $USER->id) {
|
|
$userid = $USER->id;
|
|
list($assessed, $notice) = $workshop->check_examples_assessed_before_assessment($userid);
|
|
if (!$assessed) {
|
|
throw new moodle_exception($notice, 'mod_workshop');
|
|
}
|
|
if ($workshop->phase < workshop::PHASE_ASSESSMENT) { // Can view assessments only in assessment phase onwards.
|
|
throw new moodle_exception('nopermissions', 'error', '', 'view assessments');
|
|
}
|
|
} else {
|
|
require_capability('mod/workshop:viewallassessments', $context);
|
|
$user = core_user::get_user($params['userid'], '*', MUST_EXIST);
|
|
core_user::require_active_user($user);
|
|
if (!$workshop->check_group_membership($user->id)) {
|
|
throw new moodle_exception('notingroup');
|
|
}
|
|
$userid = $user->id;
|
|
}
|
|
// Now get all my assessments (includes those pending review).
|
|
$assessmentsrecords = $workshop->get_assessments_by_reviewer($userid);
|
|
|
|
$related = array('context' => $context);
|
|
foreach ($assessmentsrecords as $assessment) {
|
|
$assessment = self::prepare_assessment_for_external($assessment, $workshop);
|
|
if (empty($assessment)) {
|
|
continue;
|
|
}
|
|
$exporter = new assessment_exporter($assessment, $related);
|
|
$assessments[] = $exporter->export($PAGE->get_renderer('core'));
|
|
}
|
|
|
|
return array(
|
|
'assessments' => $assessments,
|
|
'warnings' => $warnings
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns description of method result value
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_reviewer_assessments_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'assessments' => new external_multiple_structure(
|
|
assessment_exporter::get_read_structure()
|
|
),
|
|
'warnings' => new external_warnings()
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function update_assessment_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'assessmentid' => new external_value(PARAM_INT, 'Assessment id.'),
|
|
'data' => new external_multiple_structure (
|
|
new external_single_structure(
|
|
array(
|
|
'name' => new external_value(PARAM_ALPHANUMEXT,
|
|
'The assessment data (use WS get_assessment_form_definition for obtaining the data to sent).
|
|
Apart from that data, you can optionally send:
|
|
feedbackauthor (str); the feedback for the submission author
|
|
feedbackauthorformat (int); the format of the feedbackauthor
|
|
feedbackauthorinlineattachmentsid (int); the draft file area for the editor attachments
|
|
feedbackauthorattachmentsid (int); the draft file area id for the feedback attachments'
|
|
),
|
|
'value' => new external_value(PARAM_RAW, 'The value of the option.')
|
|
)
|
|
), 'Assessment data'
|
|
)
|
|
)
|
|
);
|
|
}
|
|
|
|
|
|
/**
|
|
* Updates an assessment.
|
|
*
|
|
* @param int $assessmentid the assessment id
|
|
* @param array $data the assessment data
|
|
* @return array indicates if the assessment was updated, the new raw grade and possible warnings.
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function update_assessment($assessmentid, $data) {
|
|
global $DB, $USER;
|
|
|
|
$params = self::validate_parameters(
|
|
self::update_assessment_parameters(), array('assessmentid' => $assessmentid, 'data' => $data)
|
|
);
|
|
$warnings = array();
|
|
|
|
// Get and validate the assessment, submission and workshop.
|
|
$assessment = $DB->get_record('workshop_assessments', array('id' => $params['assessmentid']), '*', MUST_EXIST);
|
|
$submission = $DB->get_record('workshop_submissions', array('id' => $assessment->submissionid), '*', MUST_EXIST);
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($submission->workshopid);
|
|
|
|
// Check we can edit the assessment.
|
|
$workshop->check_edit_assessment($assessment, $submission);
|
|
|
|
// Process data.
|
|
$data = new stdClass;
|
|
$data->feedbackauthor_editor = array();
|
|
|
|
foreach ($params['data'] as $wsdata) {
|
|
$name = trim($wsdata['name']);
|
|
switch ($name) {
|
|
case 'feedbackauthor':
|
|
$data->feedbackauthor_editor['text'] = $wsdata['value'];
|
|
break;
|
|
case 'feedbackauthorformat':
|
|
$data->feedbackauthor_editor['format'] = clean_param($wsdata['value'], PARAM_FORMAT);
|
|
break;
|
|
case 'feedbackauthorinlineattachmentsid':
|
|
$data->feedbackauthor_editor['itemid'] = clean_param($wsdata['value'], PARAM_INT);
|
|
break;
|
|
case 'feedbackauthorattachmentsid':
|
|
$data->feedbackauthorattachment_filemanager = clean_param($wsdata['value'], PARAM_INT);
|
|
break;
|
|
default:
|
|
$data->{$wsdata['name']} = $wsdata['value']; // Validation will be done in the form->validation.
|
|
}
|
|
}
|
|
|
|
$cansetassessmentweight = has_capability('mod/workshop:allocate', $context);
|
|
$pending = $workshop->get_pending_assessments_by_reviewer($assessment->reviewerid, $assessment->id);
|
|
// Retrieve the data from the strategy plugin.
|
|
$strategy = $workshop->grading_strategy_instance();
|
|
$mform = $strategy->get_assessment_form(null, 'assessment', $assessment, true,
|
|
array('editableweight' => $cansetassessmentweight, 'pending' => !empty($pending)));
|
|
|
|
$errors = $mform->validation((array) $data, array());
|
|
// We can get several errors, return them in warnings.
|
|
if (!empty($errors)) {
|
|
$status = false;
|
|
$rawgrade = null;
|
|
foreach ($errors as $itemname => $message) {
|
|
$warnings[] = array(
|
|
'item' => $itemname,
|
|
'itemid' => 0,
|
|
'warningcode' => 'fielderror',
|
|
'message' => s($message)
|
|
);
|
|
}
|
|
} else {
|
|
$rawgrade = $workshop->edit_assessment($assessment, $submission, $data, $strategy);
|
|
$status = true;
|
|
}
|
|
|
|
return array(
|
|
'status' => $status,
|
|
'rawgrade' => $rawgrade,
|
|
'warnings' => $warnings,
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns description of method result value
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function update_assessment_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'status' => new external_value(PARAM_BOOL, 'status: true if the assessment was added or updated false otherwise.'),
|
|
'rawgrade' => new external_value(PARAM_FLOAT, 'Raw percentual grade (0.00000 to 100.00000) for submission.',
|
|
VALUE_OPTIONAL),
|
|
'warnings' => new external_warnings()
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_grades_parameters() {
|
|
return new external_function_parameters (
|
|
array(
|
|
'workshopid' => new external_value(PARAM_INT, 'Workshop instance id.'),
|
|
'userid' => new external_value(PARAM_INT, 'User id (empty or 0 for current user).', VALUE_DEFAULT, 0),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns the grades information for the given workshop and user.
|
|
*
|
|
* @param int $workshopid workshop instance id
|
|
* @param int $userid user id
|
|
* @return array of warnings and the user plan
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function get_grades($workshopid, $userid = 0) {
|
|
global $USER;
|
|
|
|
$params = array(
|
|
'workshopid' => $workshopid,
|
|
'userid' => $userid,
|
|
);
|
|
$params = self::validate_parameters(self::get_grades_parameters(), $params);
|
|
$warnings = array();
|
|
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($params['workshopid']);
|
|
|
|
// Extra checks so only users with permissions can view other users plans.
|
|
if (empty($params['userid']) || $params['userid'] == $USER->id) {
|
|
$userid = $USER->id;
|
|
} else {
|
|
require_capability('mod/workshop:viewallassessments', $context);
|
|
$user = core_user::get_user($params['userid'], '*', MUST_EXIST);
|
|
core_user::require_active_user($user);
|
|
if (!$workshop->check_group_membership($user->id)) {
|
|
throw new moodle_exception('notingroup');
|
|
}
|
|
$userid = $user->id;
|
|
}
|
|
|
|
$finalgrades = $workshop->get_gradebook_grades($userid);
|
|
|
|
$result = array('warnings' => $warnings);
|
|
if ($finalgrades !== false) {
|
|
if (!empty($finalgrades->submissiongrade)) {
|
|
if (is_numeric($finalgrades->submissiongrade->grade)) {
|
|
$result['submissionrawgrade'] = $finalgrades->submissiongrade->grade;
|
|
}
|
|
$result['submissionlongstrgrade'] = $finalgrades->submissiongrade->str_long_grade;
|
|
$result['submissiongradehidden'] = $finalgrades->submissiongrade->hidden;
|
|
}
|
|
if (!empty($finalgrades->assessmentgrade)) {
|
|
if (is_numeric($finalgrades->assessmentgrade->grade)) {
|
|
$result['assessmentrawgrade'] = $finalgrades->assessmentgrade->grade;
|
|
}
|
|
$result['assessmentlongstrgrade'] = $finalgrades->assessmentgrade->str_long_grade;
|
|
$result['assessmentgradehidden'] = $finalgrades->assessmentgrade->hidden;
|
|
}
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* Returns description of method result value.
|
|
*
|
|
* @return external_single_structure
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_grades_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'assessmentrawgrade' => new external_value(PARAM_FLOAT, 'The assessment raw (numeric) grade.', VALUE_OPTIONAL),
|
|
'assessmentlongstrgrade' => new external_value(PARAM_NOTAGS, 'The assessment string grade.', VALUE_OPTIONAL),
|
|
'assessmentgradehidden' => new external_value(PARAM_BOOL, 'Whether the grade is hidden or not.', VALUE_OPTIONAL),
|
|
'submissionrawgrade' => new external_value(PARAM_FLOAT, 'The submission raw (numeric) grade.', VALUE_OPTIONAL),
|
|
'submissionlongstrgrade' => new external_value(PARAM_NOTAGS, 'The submission string grade.', VALUE_OPTIONAL),
|
|
'submissiongradehidden' => new external_value(PARAM_BOOL, 'Whether the grade is hidden or not.', VALUE_OPTIONAL),
|
|
'warnings' => new external_warnings(),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function evaluate_assessment_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'assessmentid' => new external_value(PARAM_INT, 'Assessment id.'),
|
|
'feedbacktext' => new external_value(PARAM_RAW, 'The feedback for the reviewer.', VALUE_DEFAULT, ''),
|
|
'feedbackformat' => new external_value(PARAM_INT, 'The feedback format for text.', VALUE_DEFAULT, FORMAT_MOODLE),
|
|
'weight' => new external_value(PARAM_INT, 'The new weight for the assessment.', VALUE_DEFAULT, 1),
|
|
'gradinggradeover' => new external_value(PARAM_ALPHANUMEXT, 'The new grading grade.', VALUE_DEFAULT, ''),
|
|
)
|
|
);
|
|
}
|
|
|
|
|
|
/**
|
|
* Evaluates an assessment (used by teachers for provide feedback to the reviewer).
|
|
*
|
|
* @param int $assessmentid the assessment id
|
|
* @param str $feedbacktext the feedback for the reviewer
|
|
* @param int $feedbackformat the feedback format for the reviewer text
|
|
* @param int $weight the new weight for the assessment
|
|
* @param mixed $gradinggradeover the new grading grade (empty for no overriding the grade)
|
|
* @return array containing the status and warnings.
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function evaluate_assessment($assessmentid, $feedbacktext = '', $feedbackformat = FORMAT_MOODLE, $weight = 1,
|
|
$gradinggradeover = '') {
|
|
global $DB;
|
|
|
|
$params = self::validate_parameters(
|
|
self::evaluate_assessment_parameters(),
|
|
array(
|
|
'assessmentid' => $assessmentid,
|
|
'feedbacktext' => $feedbacktext,
|
|
'feedbackformat' => $feedbackformat,
|
|
'weight' => $weight,
|
|
'gradinggradeover' => $gradinggradeover,
|
|
)
|
|
);
|
|
$warnings = array();
|
|
|
|
// Get and validate the assessment, submission and workshop.
|
|
$assessment = $DB->get_record('workshop_assessments', array('id' => $params['assessmentid']), '*', MUST_EXIST);
|
|
$submission = $DB->get_record('workshop_submissions', array('id' => $assessment->submissionid), '*', MUST_EXIST);
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($submission->workshopid);
|
|
|
|
// Check we can evaluate the assessment.
|
|
$workshop->check_view_assessment($assessment, $submission);
|
|
$cansetassessmentweight = has_capability('mod/workshop:allocate', $context);
|
|
$canoverridegrades = has_capability('mod/workshop:overridegrades', $context);
|
|
if (!$canoverridegrades && !$cansetassessmentweight) {
|
|
throw new moodle_exception('nopermissions', 'error', '', 'evaluate assessments');
|
|
}
|
|
|
|
// Process data.
|
|
$data = new stdClass;
|
|
$data->asid = $assessment->id;
|
|
$data->feedbackreviewer_editor = array(
|
|
'text' => $params['feedbacktext'],
|
|
'format' => $params['feedbackformat'],
|
|
);
|
|
$data->weight = $params['weight'];
|
|
$data->gradinggradeover = $params['gradinggradeover'];
|
|
|
|
$options = array(
|
|
'editable' => true,
|
|
'editableweight' => $cansetassessmentweight,
|
|
'overridablegradinggrade' => $canoverridegrades
|
|
);
|
|
$feedbackform = $workshop->get_feedbackreviewer_form(null, $assessment, $options);
|
|
|
|
$errors = $feedbackform->validation((array) $data, array());
|
|
// Extra checks for the new grade and weight.
|
|
$possibleweights = workshop::available_assessment_weights_list();
|
|
if ($data->weight < 0 || $data->weight > max(array_keys($possibleweights))) {
|
|
$errors['weight'] = 'The new weight must be higher or equal to 0 and cannot be higher than the maximum weight for
|
|
assessment.';
|
|
}
|
|
if (is_numeric($data->gradinggradeover) &&
|
|
($data->gradinggradeover < 0 || $data->gradinggradeover > $workshop->gradinggrade)) {
|
|
$errors['gradinggradeover'] = 'The new grade must be higher or equal to 0 and cannot be higher than the maximum grade
|
|
for assessment.';
|
|
}
|
|
|
|
// We can get several errors, return them in warnings.
|
|
if (!empty($errors)) {
|
|
$status = false;
|
|
foreach ($errors as $itemname => $message) {
|
|
$warnings[] = array(
|
|
'item' => $itemname,
|
|
'itemid' => 0,
|
|
'warningcode' => 'fielderror',
|
|
'message' => s($message)
|
|
);
|
|
}
|
|
} else {
|
|
$workshop->evaluate_assessment($assessment, $data, $cansetassessmentweight, $canoverridegrades);
|
|
$status = true;
|
|
}
|
|
|
|
return array(
|
|
'status' => $status,
|
|
'warnings' => $warnings,
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns description of method result value
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function evaluate_assessment_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'status' => new external_value(PARAM_BOOL, 'status: true if the assessment was evaluated, false otherwise.'),
|
|
'warnings' => new external_warnings()
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns description of method parameters
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_grades_report_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'workshopid' => new external_value(PARAM_INT, 'Workshop instance id.'),
|
|
'groupid' => new external_value(PARAM_INT, 'Group id, 0 means that the function will determine the user group.',
|
|
VALUE_DEFAULT, 0),
|
|
'sortby' => new external_value(PARAM_ALPHA, 'sort by this element: lastname, firstname, submissiontitle,
|
|
submissionmodified, submissiongrade, gradinggrade.', VALUE_DEFAULT, 'lastname'),
|
|
'sortdirection' => new external_value(PARAM_ALPHA, 'sort direction: ASC or DESC', VALUE_DEFAULT, 'ASC'),
|
|
'page' => new external_value(PARAM_INT, 'The page of records to return.', VALUE_DEFAULT, 0),
|
|
'perpage' => new external_value(PARAM_INT, 'The number of records to return per page.', VALUE_DEFAULT, 0),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Retrieves the assessment grades report.
|
|
*
|
|
* @param int $workshopid the workshop instance id
|
|
* @param int $groupid (optional) group id, 0 means that the function will determine the user group
|
|
* @param string $sortby sort by this element
|
|
* @param string $sortdirection sort direction: ASC or DESC
|
|
* @param int $page page of records to return
|
|
* @param int $perpage number of records to return per page
|
|
* @return array of warnings and the report data
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function get_grades_report($workshopid, $groupid = 0, $sortby = 'lastname', $sortdirection = 'ASC',
|
|
$page = 0, $perpage = 0) {
|
|
global $USER;
|
|
|
|
$params = array('workshopid' => $workshopid, 'groupid' => $groupid, 'sortby' => $sortby, 'sortdirection' => $sortdirection,
|
|
'page' => $page, 'perpage' => $perpage);
|
|
$params = self::validate_parameters(self::get_grades_report_parameters(), $params);
|
|
$submissions = $warnings = array();
|
|
|
|
$sortallowedvalues = array('lastname', 'firstname', 'submissiontitle', 'submissionmodified', 'submissiongrade',
|
|
'gradinggrade');
|
|
if (!in_array($params['sortby'], $sortallowedvalues)) {
|
|
throw new invalid_parameter_exception('Invalid value for sortby parameter (value: ' . $sortby . '),' .
|
|
'allowed values are: ' . implode(',', $sortallowedvalues));
|
|
}
|
|
|
|
$sortdirection = strtoupper($params['sortdirection']);
|
|
$directionallowedvalues = array('ASC', 'DESC');
|
|
if (!in_array($sortdirection, $directionallowedvalues)) {
|
|
throw new invalid_parameter_exception('Invalid value for sortdirection parameter (value: ' . $sortdirection . '),' .
|
|
'allowed values are: ' . implode(',', $directionallowedvalues));
|
|
}
|
|
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($params['workshopid']);
|
|
require_capability('mod/workshop:viewallassessments', $context);
|
|
|
|
if (!empty($params['groupid'])) {
|
|
$groupid = $params['groupid'];
|
|
// Determine is the group is visible to user.
|
|
if (!groups_group_visible($groupid, $course, $cm)) {
|
|
throw new moodle_exception('notingroup');
|
|
}
|
|
} else {
|
|
// Check to see if groups are being used here.
|
|
if ($groupmode = groups_get_activity_groupmode($cm)) {
|
|
$groupid = groups_get_activity_group($cm);
|
|
// Determine is the group is visible to user (this is particullary for the group 0 -> all groups).
|
|
if (!groups_group_visible($groupid, $course, $cm)) {
|
|
throw new moodle_exception('notingroup');
|
|
}
|
|
} else {
|
|
$groupid = 0;
|
|
}
|
|
}
|
|
|
|
if ($workshop->phase >= workshop::PHASE_SUBMISSION) {
|
|
$showauthornames = has_capability('mod/workshop:viewauthornames', $context);
|
|
$showreviewernames = has_capability('mod/workshop:viewreviewernames', $context);
|
|
|
|
if ($workshop->phase >= workshop::PHASE_EVALUATION) {
|
|
$showsubmissiongrade = true;
|
|
$showgradinggrade = true;
|
|
} else {
|
|
$showsubmissiongrade = false;
|
|
$showgradinggrade = false;
|
|
}
|
|
|
|
$data = $workshop->prepare_grading_report_data($USER->id, $groupid, $params['page'], $params['perpage'],
|
|
$params['sortby'], $sortdirection);
|
|
|
|
if (!empty($data)) {
|
|
// Populate the display options for the submissions report.
|
|
$reportopts = new stdclass();
|
|
$reportopts->showauthornames = $showauthornames;
|
|
$reportopts->showreviewernames = $showreviewernames;
|
|
$reportopts->sortby = $params['sortby'];
|
|
$reportopts->sorthow = $sortdirection;
|
|
$reportopts->showsubmissiongrade = $showsubmissiongrade;
|
|
$reportopts->showgradinggrade = $showgradinggrade;
|
|
$reportopts->workshopphase = $workshop->phase;
|
|
|
|
$report = new workshop_grading_report($data, $reportopts);
|
|
return array(
|
|
'report' => $report->export_data_for_external(),
|
|
'warnings' => array(),
|
|
);
|
|
}
|
|
}
|
|
throw new moodle_exception('nothingfound', 'workshop');
|
|
}
|
|
|
|
/**
|
|
* Returns description of method result value
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function get_grades_report_returns() {
|
|
|
|
$reviewstructure = new external_single_structure(
|
|
array(
|
|
'userid' => new external_value(PARAM_INT, 'The id of the user (0 when is configured to do not display names).'),
|
|
'assessmentid' => new external_value(PARAM_INT, 'The id of the assessment.'),
|
|
'submissionid' => new external_value(PARAM_INT, 'The id of the submission assessed.'),
|
|
'grade' => new external_value(PARAM_FLOAT, 'The grade for submission.'),
|
|
'gradinggrade' => new external_value(PARAM_FLOAT, 'The grade for assessment.'),
|
|
'gradinggradeover' => new external_value(PARAM_FLOAT, 'The aggregated grade overrided.'),
|
|
'weight' => new external_value(PARAM_INT, 'The weight of the assessment for aggregation.'),
|
|
)
|
|
);
|
|
|
|
return new external_single_structure(
|
|
array(
|
|
'report' => new external_single_structure(
|
|
array(
|
|
'grades' => new external_multiple_structure(
|
|
new external_single_structure(
|
|
array(
|
|
'userid' => new external_value(PARAM_INT, 'The id of the user being displayed in the report.'),
|
|
'submissionid' => new external_value(PARAM_INT, 'Submission id.'),
|
|
'submissiontitle' => new external_value(PARAM_RAW, 'Submission title.'),
|
|
'submissionmodified' => new external_value(PARAM_INT, 'Timestamp submission was updated.'),
|
|
'submissiongrade' => new external_value(PARAM_FLOAT, 'Aggregated grade for the submission.',
|
|
VALUE_OPTIONAL),
|
|
'gradinggrade' => new external_value(PARAM_FLOAT, 'Computed grade for the assessment.',
|
|
VALUE_OPTIONAL),
|
|
'submissiongradeover' => new external_value(PARAM_FLOAT, 'Grade for the assessment overrided
|
|
by the teacher.', VALUE_OPTIONAL),
|
|
'submissiongradeoverby' => new external_value(PARAM_INT, 'The id of the user who overrided
|
|
the grade.', VALUE_OPTIONAL),
|
|
'submissionpublished' => new external_value(PARAM_INT, 'Whether is a submission published.',
|
|
VALUE_OPTIONAL),
|
|
'reviewedby' => new external_multiple_structure($reviewstructure, 'The users who reviewed the
|
|
user submission.', VALUE_OPTIONAL),
|
|
'reviewerof' => new external_multiple_structure($reviewstructure, 'The assessments the user
|
|
reviewed.', VALUE_OPTIONAL),
|
|
)
|
|
)
|
|
),
|
|
'totalcount' => new external_value(PARAM_INT, 'Number of total submissions.'),
|
|
)
|
|
),
|
|
'warnings' => new external_warnings()
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Describes the parameters for view_submission.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function view_submission_parameters() {
|
|
return new external_function_parameters (
|
|
array(
|
|
'submissionid' => new external_value(PARAM_INT, 'Submission id'),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Trigger the submission viewed event.
|
|
*
|
|
* @param int $submissionid submission id
|
|
* @return array of warnings and status result
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function view_submission($submissionid) {
|
|
global $DB;
|
|
|
|
$params = self::validate_parameters(self::view_submission_parameters(), array('submissionid' => $submissionid));
|
|
$warnings = array();
|
|
|
|
// Get and validate the submission and workshop.
|
|
$submission = $DB->get_record('workshop_submissions', array('id' => $params['submissionid']), '*', MUST_EXIST);
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($submission->workshopid);
|
|
|
|
self::validate_submission($submission, $workshop);
|
|
|
|
$workshop->set_submission_viewed($submission);
|
|
|
|
$result = array(
|
|
'status' => true,
|
|
'warnings' => $warnings,
|
|
);
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* Describes the view_submission return value.
|
|
*
|
|
* @return external_single_structure
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function view_submission_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'status' => new external_value(PARAM_BOOL, 'status: true if success'),
|
|
'warnings' => new external_warnings(),
|
|
)
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns the description of the external function parameters.
|
|
*
|
|
* @return external_function_parameters
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function evaluate_submission_parameters() {
|
|
return new external_function_parameters(
|
|
array(
|
|
'submissionid' => new external_value(PARAM_INT, 'submission id.'),
|
|
'feedbacktext' => new external_value(PARAM_RAW, 'The feedback for the author.', VALUE_DEFAULT, ''),
|
|
'feedbackformat' => new external_value(PARAM_INT, 'The feedback format for text.', VALUE_DEFAULT, FORMAT_MOODLE),
|
|
'published' => new external_value(PARAM_BOOL, 'Publish the submission for others?.', VALUE_DEFAULT, false),
|
|
'gradeover' => new external_value(PARAM_ALPHANUMEXT, 'The new submission grade.', VALUE_DEFAULT, ''),
|
|
)
|
|
);
|
|
}
|
|
|
|
|
|
/**
|
|
* Evaluates a submission (used by teachers for provide feedback or override the submission grade).
|
|
*
|
|
* @param int $submissionid the submission id
|
|
* @param str $feedbacktext the feedback for the author
|
|
* @param int $feedbackformat the feedback format for the reviewer text
|
|
* @param bool $published whether to publish the submission for other users
|
|
* @param mixed $gradeover the new submission grade (empty for no overriding the grade)
|
|
* @return array containing the status and warnings.
|
|
* @since Moodle 3.4
|
|
* @throws moodle_exception
|
|
*/
|
|
public static function evaluate_submission($submissionid, $feedbacktext = '', $feedbackformat = FORMAT_MOODLE, $published = 1,
|
|
$gradeover = '') {
|
|
global $DB;
|
|
|
|
$params = self::validate_parameters(
|
|
self::evaluate_submission_parameters(),
|
|
array(
|
|
'submissionid' => $submissionid,
|
|
'feedbacktext' => $feedbacktext,
|
|
'feedbackformat' => $feedbackformat,
|
|
'published' => $published,
|
|
'gradeover' => $gradeover,
|
|
)
|
|
);
|
|
$warnings = array();
|
|
|
|
// Get and validate the submission, submission and workshop.
|
|
$submission = $DB->get_record('workshop_submissions', array('id' => $params['submissionid']), '*', MUST_EXIST);
|
|
list($workshop, $course, $cm, $context) = self::validate_workshop($submission->workshopid);
|
|
|
|
// Check we can evaluate the submission.
|
|
self::validate_submission($submission, $workshop);
|
|
$canpublish = has_capability('mod/workshop:publishsubmissions', $context);
|
|
$canoverride = ($workshop->phase == workshop::PHASE_EVALUATION &&
|
|
has_capability('mod/workshop:overridegrades', $context));
|
|
|
|
if (!$canpublish && !$canoverride) {
|
|
throw new moodle_exception('nopermissions', 'error', '', 'evaluate submission');
|
|
}
|
|
|
|
// Process data.
|
|
$data = new stdClass;
|
|
$data->id = $submission->id;
|
|
$data->feedbackauthor_editor = array(
|
|
'text' => $params['feedbacktext'],
|
|
'format' => $params['feedbackformat'],
|
|
);
|
|
$data->published = $params['published'];
|
|
$data->gradeover = $params['gradeover'];
|
|
|
|
$options = array(
|
|
'editable' => true,
|
|
'editablepublished' => $canpublish,
|
|
'overridablegrade' => $canoverride
|
|
);
|
|
$feedbackform = $workshop->get_feedbackauthor_form(null, $submission, $options);
|
|
|
|
$errors = $feedbackform->validation((array) $data, array());
|
|
// Extra checks for the new grade (if set).
|
|
if (is_numeric($data->gradeover) && $data->gradeover > $workshop->grade) {
|
|
$errors['gradeover'] = 'The new grade cannot be higher than the maximum grade for submission.';
|
|
}
|
|
|
|
// We can get several errors, return them in warnings.
|
|
if (!empty($errors)) {
|
|
$status = false;
|
|
foreach ($errors as $itemname => $message) {
|
|
$warnings[] = array(
|
|
'item' => $itemname,
|
|
'itemid' => 0,
|
|
'warningcode' => 'fielderror',
|
|
'message' => s($message)
|
|
);
|
|
}
|
|
} else {
|
|
$workshop->evaluate_submission($submission, $data, $canpublish, $canoverride);
|
|
$status = true;
|
|
}
|
|
|
|
return array(
|
|
'status' => $status,
|
|
'warnings' => $warnings,
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns description of method result value
|
|
*
|
|
* @return external_description
|
|
* @since Moodle 3.4
|
|
*/
|
|
public static function evaluate_submission_returns() {
|
|
return new external_single_structure(
|
|
array(
|
|
'status' => new external_value(PARAM_BOOL, 'status: true if the submission was evaluated, false otherwise.'),
|
|
'warnings' => new external_warnings()
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|