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() | |
|             ) | |
|         ); | |
|     } | |
| }
 | |
| 
 |