. /** * Webservice class for managing studyplans * @package local_treestudyplan * @copyright 2023 P.M. Kuipers * @license https://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ namespace local_treestudyplan; defined('MOODLE_INTERNAL') || die(); use local_treestudyplan\local\helpers\webservicehelper; require_once($CFG->libdir.'/externallib.php'); require_once($CFG->libdir.'/badgeslib.php'); require_once($CFG->libdir.'/gradelib.php'); require_once($CFG->dirroot.'/course/modlib.php'); /** * Webservice class for managing studyplans */ class studyplanservice extends \external_api { /** * Capability required to edit study plans * @var string */ const CAP_EDIT = "local/treestudyplan:editstudyplan"; /** * Capability required to view studyplans (for other users) * @var string */ const CAP_VIEW = "local/treestudyplan:viewuserreports"; /************************ * * * list_studyplans * * * ************************/ /** * Parameter description for webservice function list_studyplans */ public static function list_studyplans_parameters() : \external_function_parameters { return new \external_function_parameters([ "context_id" => new \external_value(PARAM_INT, 'context to search in for studyplans', VALUE_DEFAULT), ]); } /** * Return value description for webservice function list_studyplans */ public static function list_studyplans_returns() : \external_description { return new \external_multiple_structure( studyplan::simple_structure() ); } /** * Get overview of all studyplans in a given context * @param int $contextid Id of context to view (defaults to systemcontext) * @return array */ public static function list_studyplans($contextid = 0) { global $CFG, $DB; // Check if the user has the correct permissions for this context. $context = webservicehelper::find_context($contextid); webservicehelper::require_capabilities([self::CAP_EDIT, self::CAP_VIEW], $context); // Now list all the studyplans in the relevant context. $list = []; $studyplans = studyplan::find_all($contextid); foreach ($studyplans as $studyplan) { $list[] = $studyplan->simple_model(); } return $list; } /************************ * * * get_studyplan_map * * * ************************/ /** * Parameter description for webservice function get_studyplan_map */ public static function get_studyplan_map_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of a studyplan to check usage on', VALUE_REQUIRED), ] ); } /** * Return value description for webservice function get_studyplan_map */ public static function get_studyplan_map_returns() : \external_description { return studyplan::editor_structure(); } /** * Get editor model for specific studyplan * @param int $id Id of studyplan * @return array */ public static function get_studyplan_map($id) { if (isset($id) && $id > 0) { $studyplan = studyplan::find_by_id($id); webservicehelper::require_capabilities([self::CAP_EDIT, self::CAP_VIEW], $studyplan->context()); return $studyplan->editor_model(); } else { return null; } } /************************ * * * get_studyline_map * * * ************************/ /** * Parameter description for webservice function get_studyline_map */ public static function get_studyline_map_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of a studyline to check usage on', VALUE_DEFAULT), ]); } /** * Return value description for webservice function get_studyline_map */ public static function get_studyline_map_returns() : \external_description { return new \external_multiple_structure( studyline::editor_structure() ); } /** * Get editor model for specific study line * @param int $id ID of study line * @return array */ public static function get_studyline_map($id) { $o = studyline::find_by_id($id); webservicehelper::require_capabilities([self::CAP_EDIT, self::CAP_VIEW], $o->context()); return $o->editor_model(); } /************************ * * * add_studyplan * * * ************************/ /** * Parameter description for webservice function add_studyplan */ public static function add_studyplan_parameters() : \external_function_parameters { return new \external_function_parameters( [ "name" => new \external_value(PARAM_TEXT, 'name of studyplan'), "shortname" => new \external_value(PARAM_TEXT, 'shortname of studyplan'), "idnumber" => new \external_value(PARAM_TEXT, 'idnumber of studyplan'), "description" => new \external_value(PARAM_TEXT, 'description of studyplan'), "periods" => new \external_value(PARAM_INT, 'number of periods in studyplan'), "startdate" => new \external_value(PARAM_TEXT, 'start date of studyplan'), "enddate" => new \external_value(PARAM_TEXT, 'end date of studyplan'), "aggregation" => new \external_value(PARAM_TEXT, 'selected aggregator', VALUE_DEFAULT), "aggregation_config" => new \external_value(PARAM_TEXT, 'config string for aggregator', VALUE_DEFAULT), "context_id" => new \external_value(PARAM_INT, 'context of the study plan', VALUE_DEFAULT), ] ); } /** * Return value description for webservice function add_studyplan */ public static function add_studyplan_returns() : \external_description { return studyplan::simple_structure(); } /** * Add a new studyplan in a given context * @param mixed $name * @param mixed $shortname * @param mixed $idnumber * @param mixed $description * @param mixed $periods * @param mixed $startdate * @param mixed $enddate * @param string $aggregation * @param string $aggregationconfig * @param int $contextid * @return array */ public static function add_studyplan($name, $shortname, $idnumber, $description, $periods, $startdate, $enddate, $aggregation = "bistate", $aggregationconfig = '', $contextid = 0) { // Check if we have the proper rights for the requested context. $context = webservicehelper::find_context($contextid); webservicehelper::require_capabilities(self::CAP_EDIT, $context); $o = studyplan::add([ 'name' => $name, 'shortname' => $shortname, 'idnumber' => $idnumber, 'description' => $description, 'periods' => $periods, 'startdate' => $startdate, 'enddate' => empty($enddate) ? null : $enddate, 'aggregation' => $aggregation, 'aggregation_config' => $aggregationconfig, 'context_id' => $contextid, ]); return $o->simple_model(); } /************************ * * * edit_studyplan * * * ************************/ /** * Parameter description for webservice function edit_studyplan */ public static function edit_studyplan_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of studyplan'), "name" => new \external_value(PARAM_TEXT, 'name of studyplan'), "shortname" => new \external_value(PARAM_TEXT, 'shortname of studyplan'), "idnumber" => new \external_value(PARAM_TEXT, 'idnumber of studyplan'), "description" => new \external_value(PARAM_TEXT, 'description of studyplan'), "periods" => new \external_value(PARAM_INT, 'number of periods in studyplan'), "startdate" => new \external_value(PARAM_TEXT, 'start date of studyplan'), "enddate" => new \external_value(PARAM_TEXT, 'end date of studyplan'), "aggregation" => new \external_value(PARAM_TEXT, 'selected aggregator', VALUE_DEFAULT), "aggregation_config" => new \external_value(PARAM_TEXT, 'config string for aggregator', VALUE_DEFAULT), "context_id" => new \external_value(PARAM_INT, 'context of the study plan', VALUE_DEFAULT), ] ); } /** * Return value description for webservice function edit_studyplan */ public static function edit_studyplan_returns() : \external_description { return studyplan::simple_structure(); } /** * Edit studyplan parameters * @param mixed $id * @param mixed $name * @param mixed $shortname * @param mixed $idnumber * @param mixed $description * @param mixed $periods * @param mixed $startdate * @param mixed $enddate * @param string $aggregation * @param string $aggregationconfig * @param int $contextid * @return array */ public static function edit_studyplan($id, $name, $shortname, $idnumber, $description, $periods, $startdate, $enddate, $aggregation = "bistate", $aggregationconfig = '', $contextid = 0) { // Validate access in the intended context. $context = webservicehelper::find_context($contextid); // Do not validate the context in this case, just check the permissions. webservicehelper::require_capabilities(self::CAP_EDIT, $context, false); $o = studyplan::find_by_id($id); webservicehelper::require_capabilities(self::CAP_EDIT, $o->context()); $o->edit([ 'name' => $name, 'shortname' => $shortname, 'idnumber' => $idnumber, 'description' => $description, 'periods' => $periods, 'startdate' => $startdate, 'enddate' => $enddate, 'aggregation' => $aggregation, 'aggregation_config' => $aggregationconfig, 'context_id' => $contextid, ]); return $o->simple_model(); } /************************ * * * delete_studyplan * * * ************************/ /** * Parameter description for webservice function delete_studyplan */ public static function delete_studyplan_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of studyplan'), "force" => new \external_value(PARAM_BOOL, 'id of studyplan', VALUE_DEFAULT), ] ); } /** * Return value description for webservice function delete_studyplan */ public static function delete_studyplan_returns() : \external_description { return success::structure(); } /** * Delete a studyplan * @param mixed $id Id of the studyplan * @param bool $force Force deletion, even though studyplan is not empty * @return array Succes/fail model */ public static function delete_studyplan($id, $force = false) { $o = studyplan::find_by_id($id); // Validate if the requesting user has the right to edit the plan in it's current context. webservicehelper::require_capabilities(self::CAP_EDIT, $o->context()); return $o->delete(!!$force)->model(); } /************************ * * * add_studyline * * * ************************/ /** * Parameter description for webservice function add_studyline */ public static function add_studyline_parameters() : \external_function_parameters { return new \external_function_parameters( [ "page_id" => new \external_value(PARAM_INT, 'id of studyplan to add line to'), "name" => new \external_value(PARAM_TEXT, 'shortname of studyline'), "shortname" => new \external_value(PARAM_TEXT, 'idnumber of studyline'), "color" => new \external_value(PARAM_TEXT, 'description of studyline'), "sequence" => new \external_value(PARAM_INT, 'sequence of studyline'), ] ); } /** * Return value description for webservice function add_studyline */ public static function add_studyline_returns() : \external_description { return studyline::editor_structure(); } /** * Add a new study line * @param mixed $pageid * @param mixed $name * @param mixed $shortname * @param mixed $color * @param mixed $sequence * @return array */ public static function add_studyline($pageid, $name, $shortname, $color, $sequence) { // Validate if the requesting user has the right to edit the plan in it's current context. $page = studyplanpage::find_by_id($pageid); webservicehelper::require_capabilities(self::CAP_EDIT, $page->studyplan()->context()); $o = studyline::add([ 'page_id' => $pageid, 'name' => $name, 'shortname' => $shortname, 'color' => $color, 'sequence' => $sequence, ]); return $o->editor_model(); } /************************ * * * edit_studyline * * * ************************/ /** * Parameter description for webservice function edit_studyline */ public static function edit_studyline_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of studyline'), "name" => new \external_value(PARAM_TEXT, 'shortname of studyline'), "shortname" => new \external_value(PARAM_TEXT, 'idnumber of studyline'), "color" => new \external_value(PARAM_TEXT, 'description of studyline'), ] ); } /** * Return value description for webservice function edit_studyline */ public static function edit_studyline_returns() : \external_description { return studyline::editor_structure(); } /** * Edit studyline parameters * @param mixed $id * @param mixed $name * @param mixed $shortname * @param mixed $color * @return [type] */ public static function edit_studyline($id, $name, $shortname, $color) { $o = studyline::find_by_id($id); // Validate if the requesting user has the right to edit the plan in it's current context. webservicehelper::require_capabilities(self::CAP_EDIT, $o->context()); $o->edit([ 'name' => $name, 'shortname' => $shortname, 'color' => $color, ]); return $o->editor_model(); } /************************ * * * delete_studyline * * * ************************/ /** * Parameter description for webservice function delete_studyline */ public static function delete_studyline_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of studyline'), ] ); } /** * Return value description for webservice function delete_studyline */ public static function delete_studyline_returns() : \external_description { return success::structure(); } /** * Delete a study line * @param mixed $id Id of the studyline * @return array Success/fail model * */ public static function delete_studyline($id) { $o = studyline::find_by_id($id); // Validate if the requesting user has the right to edit the plan in it's current context. webservicehelper::require_capabilities(self::CAP_EDIT, $o->context()); return $o->delete()->model(); } /************************ * * * reorder_studylines * * * ************************/ /** * Parameter description for webservice function reorder_studylines */ public static function reorder_studylines_parameters() : \external_function_parameters { return new \external_function_parameters( [ "sequence" => new \external_multiple_structure( new \external_single_structure([ "id" => new \external_value(PARAM_INT, 'id of studyline '), "sequence" => new \external_value(PARAM_INT, 'order of studyline'), ]), ), ] ); } /** * Return value description for webservice function reorder_studylines */ public static function reorder_studylines_returns() : \external_description { return success::structure(); } /** * Reorder study lines * @param int[] $resequence New order of study lines by id * @return array */ public static function reorder_studylines($resequence) { // Validate if the requesting user has the right to edit the lines in it's current context. foreach ($resequence as $sq) { $o = studyline::find_by_id(($sq['id'])); webservicehelper::require_capabilities(self::CAP_EDIT, $o->context()); } return studyline::reorder($resequence)->model(); } /*********************************** * STUDYITEM FUNCTIONS ***********************************/ /************************ * * * get_studyitem * * * ************************/ /** * Parameter description for webservice function get_studyitem */ public static function get_studyitem_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of a study item to retrieve'), ] ); } /** * Return value description for webservice function get_studyitem */ public static function get_studyitem_returns() : \external_description { return studyitem::editor_structure(); } /** * Get editor model for study item * @param mixed $id ID of study item * @return array */ public static function get_studyitem($id) { $o = studyitem::find_by_id($id); webservicehelper::require_capabilities([self::CAP_EDIT, self::CAP_VIEW], $o->context()); return $o->editor_model(); } /************************ * * * add_studyitem * * * ************************/ /** * Parameter description for webservice function add_studyitem */ public static function add_studyitem_parameters() : \external_function_parameters { return new \external_function_parameters( [ "line_id" => new \external_value(PARAM_INT, 'id of related study line'), "type" => new \external_value(PARAM_TEXT, 'type of study item'), "details" => new \external_single_structure([ "conditions" => new \external_value(PARAM_TEXT, 'conditions for completion', VALUE_OPTIONAL), "competency_id" => new \external_value(PARAM_INT, 'id of referenced competency', VALUE_OPTIONAL), "course_id" => new \external_value(PARAM_INT, 'id of referenced course', VALUE_OPTIONAL), "badge_id" => new \external_value(PARAM_INT, 'id of referenced badge', VALUE_OPTIONAL), "continuation_id" => new \external_value(PARAM_INT, 'id of continued item', VALUE_OPTIONAL), ]), "slot" => new \external_value(PARAM_INT, 'slot in the study plan', VALUE_DEFAULT), "layer" => new \external_value(PARAM_INT, 'layer in the slot', VALUE_OPTIONAL), ] ); } /** * Return value description for webservice function add_studyitem */ public static function add_studyitem_returns() : \external_description { return studyitem::editor_structure(); } /** * Add a new study item * @param int $lineid * @param string $type * @param array $details * @param int $slot * @param int $layer * @return array */ public static function add_studyitem($lineid, $type, $details, $slot = -1, $layer = 0) { webservicehelper::require_capabilities(self::CAP_EDIT, studyline::find_by_id($lineid)->context()); $o = studyitem::add([ 'line_id' => $lineid, 'type' => $type, 'slot' => $slot, 'layer' => $layer, 'competency_id' => isset($details['competency_id']) ? $details['competency_id'] : null, 'course_id' => isset($details['course_id']) ? $details['course_id'] : null, 'badge_id' => isset($details['badge_id']) ? $details['badge_id'] : null, 'continuation_id' => isset($details['continuation_id']) ? $details['continuation_id'] : null, ]); return $o->editor_model(); } /************************ * * * edit_studyitem * * * ************************/ /** * Parameter description for webservice function edit_studyitem */ public static function edit_studyitem_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of study item'), "conditions" => new \external_value(PARAM_TEXT, 'conditions for completion'), "continuation_id" => new \external_value(PARAM_INT, 'id of continued item', VALUE_DEFAULT), ]); } /** * Return value description for webservice function edit_studyitem */ public static function edit_studyitem_returns() : \external_description { return studyitem::editor_structure(); } /** * Edit study item paremeters * @param mixed $id Id of studt item * @param mixed $conditions Conditions related to item (filters only) * @param bool $continuationid Link to previous filter item to copy result from (not used) * @return array */ public static function edit_studyitem($id, $conditions, $continuationid = false) { $o = studyitem::find_by_id($id); webservicehelper::require_capabilities(self::CAP_EDIT, $o->context()); $config = [ 'conditions' => $conditions, ]; if ($continuationid !== false) { $config['continuation_id'] = $continuationid; } $o->edit($config); return $o->editor_model(); } /************************ * * * reorder_studyitems * * * ************************/ /** * Parameter description for webservice function reorder_studyitems */ public static function reorder_studyitems_parameters() : \external_function_parameters { return new \external_function_parameters( [ "items" => new \external_multiple_structure( new \external_single_structure([ "id" => new \external_value(PARAM_INT, 'id of studyitem '), "line_id" => new \external_value(PARAM_INT, 'id of related study line'), "slot" => new \external_value(PARAM_INT, 'slot in the study plan'), "layer" => new \external_value(PARAM_INT, 'layer in the slot'), ]), ), ] ); } /** * Return value description for webservice function reorder_studyitems */ public static function reorder_studyitems_returns() : \external_description { return success::structure(); } /** * Reposition study items in line, layer and/or slot * @param mixed $resequence Array of item info [id, line_id, slot, layer] * @return array Success/fail model */ public static function reorder_studyitems($resequence) { // Check for permissions to modify the studyplan. foreach ($resequence as $sq) { webservicehelper::require_capabilities(self::CAP_EDIT, studyitem::find_by_id(($sq['id']))->context()); } return studyitem::reorder($resequence)->model(); } /************************ * * * delete_studyitem * * * ************************/ /** * Parameter description for webservice function delete_studyitem */ public static function delete_studyitem_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of studyitem'), ] ); } /** * Return value description for webservice function delete_studyitem */ public static function delete_studyitem_returns() : \external_description { return success::structure(); } /** * Delete a studyitem * @param mixed $id Id of study item to delete * @return array Success/fail model */ public static function delete_studyitem($id) { $o = studyitem::find_by_id($id); webservicehelper::require_capabilities(self::CAP_EDIT, $o->context()); return $o->delete()->model(); } /************************ * * * connect_studyitems * * * ************************/ /** * Parameter description for webservice function connect_studyitems */ public static function connect_studyitems_parameters() : \external_function_parameters { return new \external_function_parameters( [ "from_id" => new \external_value(PARAM_INT, 'id of studyitem connect start '), "to_id" => new \external_value(PARAM_INT, 'id ofstudyitem connect end'), ] ); } /** * Return value description for webservice function connect_studyitems */ public static function connect_studyitems_returns() : \external_description { return studyitemconnection::structure(); } /** * Connect two studylines * @param mixed $fromid Originating item * @param mixed $toid Target item * @return array Success/fail model */ public static function connect_studyitems($fromid, $toid) { // Validate permissions. webservicehelper::require_capabilities(self::CAP_EDIT, studyitem::find_by_id($fromid)->context()); webservicehelper::require_capabilities(self::CAP_EDIT, studyitem::find_by_id($toid)->context()); $o = studyitemconnection::connect($fromid, $toid); return $o->model(); } /**************************** * * * disconnect_studyitems * * * ****************************/ /** * Parameter description for webservice function disconnect_studyitems */ public static function disconnect_studyitems_parameters() : \external_function_parameters { return new \external_function_parameters( [ "from_id" => new \external_value(PARAM_INT, 'id of studyitem '), "to_id" => new \external_value(PARAM_INT, 'id of related study line'), ] ); } /** * Return value description for webservice function disconnect_studyitems */ public static function disconnect_studyitems_returns() : \external_description { return success::structure(); } /** * Disconnect two studylines * @param mixed $fromid Originating item * @param mixed $toid Target item * @return array Success/fail model */ public static function disconnect_studyitems($fromid, $toid) { // Validate permissions. webservicehelper::require_capabilities(self::CAP_EDIT, studyitem::find_by_id($fromid)->context()); webservicehelper::require_capabilities(self::CAP_EDIT, studyitem::find_by_id($toid)->context()); return studyitemconnection::disconnect($fromid, $toid)->model(); } /**************************** * * * list badges * * * ****************************/ /** * Parameter description for webservice function list_badges */ public static function list_badges_parameters() : \external_function_parameters { return new \external_function_parameters( [] ); } /** * Return value description for webservice function list_badges */ public static function list_badges_returns() : \external_description { return new \external_multiple_structure(badgeinfo::editor_structure()); } /** * List all available badges to drag into a studyplan page * @return array */ public static function list_badges() { $systemcontext = webservicehelper::system_context(); $result = []; $badges = badges_get_badges(BADGE_TYPE_SITE, "timemodified"); foreach ($badges as $badge) { // TODO: Add config option to list only active badges. $result[] = (new badgeinfo($badge))->editor_model(); // TODO: Include course badges somehow... Just site badges is not enough. } return $result; } /**************************** * * * include/remove grades * * * ****************************/ /** * Parameter description for webservice function include_grade */ public static function include_grade_parameters() : \external_function_parameters { return new \external_function_parameters( [ "grade_id" => new \external_value(PARAM_INT, 'id of gradeitem '), "item_id" => new \external_value(PARAM_INT, 'id of studyitem '), "include" => new \external_value(PARAM_BOOL, 'include or not '), "required" => new \external_value(PARAM_BOOL, 'required grade or not', VALUE_DEFAULT), ] ); } /** * Return value description for webservice function include_grade */ public static function include_grade_returns() : \external_description { return success::structure(); } /** * Mark a gradable item for inclusion in the studyplan * @param mixed $gradeid Id of gradable * @param mixed $itemid Id of study item * @param bool $include Include grade or not * @param bool $required Mark grade as required or not * @return array Success/Fail model * */ public static function include_grade($gradeid, $itemid, $include, $required = false) { global $USER; // Find related course and course context. $coursecontext = gradeinfo::get_coursecontext_by_id($gradeid); // Do sanity checks. \external_api::validate_context($coursecontext); // Check correct capabilities. if (has_capability('local/treestudyplan:editstudyplan', studyitem::find_by_id($itemid)->context()) || is_enrolled($coursecontext, $USER, 'local/treestudyplan:selectowngradables')) { return gradeinfo::include_grade($gradeid, $itemid, $include, $required)->model(); } else { return success::fail("Access denied")->model(); } } /**************************************** * * * mark/unmark competency required * * * ****************************************/ /** * Parameter description for webservice function require_competency */ public static function require_competency_parameters() : \external_function_parameters { return new \external_function_parameters( [ "competency_id" => new \external_value(PARAM_INT, 'id of competency '), "item_id" => new \external_value(PARAM_INT, 'id of studyitem '), "required" => new \external_value(PARAM_BOOL, 'required grade or not', VALUE_DEFAULT), ] ); } /** * Return value description for webservice function require_competency */ public static function require_competency_returns() : \external_description { return success::structure(); } /** * Mark a competency as required for course completion * @param mixed $gradeid Id of gradable * @param mixed $itemid Id of study item * @param bool $required Mark grade as required or not * @return array Success/Fail model * */ public static function require_competency($competencyid, $itemid, $required) { global $USER; $item = studyitem::find_by_id($itemid); // Find related course and course context. if($item->courseid()) { $coursecontext = \context_course::instance($item->courseid()); // Do sanity checks. \external_api::validate_context($coursecontext); } else { $coursecontext = null; \external_api::validate_context($item->context()); } // Check correct capabilities. if (has_capability('local/treestudyplan:editstudyplan', $item->context()) || ($coursecontext && is_enrolled($coursecontext, $USER, 'local/treestudyplan:selectowngradables'))) { return coursecompetencyinfo::require_competency($competencyid, $itemid, $required)->model(); } else { return success::fail("Access denied")->model(); } } /**************************** * * * list aggregators * * * ****************************/ /** * Parameter description for webservice function list_aggregators */ public static function list_aggregators_parameters() : \external_function_parameters { return new \external_function_parameters([]); } /** * Return value description for webservice function list_aggregators */ public static function list_aggregators_returns() : \external_description { return aggregator::list_structure(); } /** * List available aggregators * @return array */ public static function list_aggregators() { return aggregator::list_model(); } /**************************** * * * force_studyplan_scale * * * ****************************/ /** * Parameter description for webservice function force_studyplan_scale */ public static function force_studyplan_scale_parameters() : \external_function_parameters { return new \external_function_parameters( [ "studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan'), "scale_id" => new \external_value(PARAM_INT, 'scale_id to set'), ] ); } /** * Return value description for webservice function force_studyplan_scale */ public static function force_studyplan_scale_returns() : \external_description { return new \external_multiple_structure(new \external_single_structure([ "course" => courseinfo::simple_structure(), "grades" => new \external_multiple_structure(new \external_single_structure([ "name" => new \external_value(PARAM_TEXT, 'grade name'), "changed" => new \external_value(PARAM_TEXT, 'changed or not'), "debug" => new \external_value(PARAM_TEXT, 'debug', VALUE_OPTIONAL), ])), ])); } /** * Force all gradables in the studyplan to the same scale item * @param int $studyplanid Id of studyplan * @param int $scaleid Id of scale to use * @return array */ public static function force_studyplan_scale($studyplanid, $scaleid) { global $DB; $dbman = $DB->get_manager(); // Validate permissions. webservicehelper::require_capabilities(self::CAP_EDIT, studyplan::find_by_id($studyplanid)->context()); $list = []; // Check if scaleid is valid. $scale = \grade_scale::fetch(['id' => $scaleid]); $scale->load_items(); if ($scale) { $gradecfg = $DB->get_record("local_treestudyplan_gradecfg", ["scale_id" => $scale->id]); $scalepass = ($gradecfg) ? $gradecfg->min_completed : 0; $scalemax = count($scale->scale_items); // Find studyline id's. $studylineids = $DB->get_fieldset_select(studyline::TABLE, "id", "studyplan_id = :plan_id", ['plan_id' => $studyplanid]); foreach ($studylineids as $studylineid) { // Find id's of studyitems of type course. $records = $DB->get_records(studyitem::TABLE, ['line_id' => $studylineid]); foreach ($records as $itemr) { $studyitem = new studyitem($itemr->id); if ($studyitem->valid() && $studyitem->type() == studyitem::COURSE) { $courseinfo = $studyitem->getcourseinfo(); $gradables = gradeinfo::list_studyitem_gradables($studyitem); $gradelist = []; foreach ($gradables as $g) { $gi = $g->get_gradeitem(); // Only change items that do not yet have grades. // Otherwise we will need to implement grade recalculations and it is not worth the trouble. . // If grades are given, you likely don't want to change it like this anyway. if (!$gi->has_grades()) { $gi->gradetype = GRADE_TYPE_SCALE; $gi->scaleid = $scale->id; $gi->grademin = 1; $gi->grademax = $scalemax; $gi->gradepass = $scalepass; // Update, signalling with our signature and bulkupdate. $result = $gi->update("local/treestudyplan"); $debug = ""; if ($result) { $updated = "converted"; } else { $updated = "error"; } // Next update the activity's table if it has a grade field. // Grade is generally set to the negative scale id if it is a scale. $tablename = $gi->itemmodule; $fieldname = "grade"; if ($result && $gi->itemtype == "mod" && $dbman->table_exists($tablename)) { if ($dbman->field_exists($tablename, $fieldname)) { $gradevalue = intval(0 - ($scale->id)); try { $DB->set_field($tablename, $fieldname, $gradevalue, ["id" => $gi->iteminstance]); } catch (\dml_exception $x) { $updated = "fail"; $debug = strval($x); } } } } else { $updated = "skipped"; } $gradelist[] = [ 'name' => $gi->itemname, 'changed' => $updated, 'debug' => $debug, ]; } $list[] = [ 'course' => $courseinfo->simple_model(), 'grades' => $gradelist, ]; } } } } return $list; } /**************************** * * * list_scales * * * ****************************/ /** * Parameter description for webservice function list_scales */ public static function list_scales_parameters() : \external_function_parameters { return new \external_function_parameters( [] ); } /** * Return value description for webservice function list_scales */ public static function list_scales_returns() : \external_description { return new \external_multiple_structure(new \external_single_structure([ "id" => new \external_value(PARAM_INT, 'id of scale'), "name" => new \external_value(PARAM_TEXT, 'scale name'), ])); } /** * List available scales * @return array */ public static function list_scales() { global $DB; $list = []; $scales = \grade_scale::fetch_all_global(); foreach ($scales as $scale) { $list[] = [ "id" => $scale->id, "name" => $scale->name, ]; } return $list; } /**************************** * * * duplicate studyplan * * * ****************************/ /** * Parameter description for webservice function duplicate_plan */ public static function duplicate_plan_parameters() : \external_function_parameters { return new \external_function_parameters( [ "plan_id" => new \external_value(PARAM_INT, 'id of plan to copy '), "name" => new \external_value(PARAM_TEXT, 'name of copy '), "shortname" => new \external_value(PARAM_TEXT, 'shortname of copy '), ] ); } /** * Return value description for webservice function duplicate_plan */ public static function duplicate_plan_returns() : \external_description { return studyplan::simple_structure(); } /** * Duplicate a studyplan into a new one * @param mixed $studyplanid Id of the plan to duplicate * @param mixed $name New fullname * @param mixed $shortname New shortname * @return array */ public static function duplicate_plan($studyplanid, $name, $shortname) { // Validate permissions. webservicehelper::require_capabilities(self::CAP_EDIT, studyplan::find_by_id($studyplanid)->context()); return studyplan::duplicate_plan($studyplanid, $name, $shortname); } /**************************** * * * export studyplan * * * ****************************/ /** * Parameter description for webservice function export_plan */ public static function export_plan_parameters() : \external_function_parameters { return new \external_function_parameters( [ "studyplan_id" => new \external_value(PARAM_INT, 'id of plan to export '), "format" => new \external_value(PARAM_TEXT, 'export format', VALUE_OPTIONAL), ] ); } /** * Return value description for webservice function export_plan */ public static function export_plan_returns() : \external_description { return studyplan::export_structure(); } /** * Export studyplan * @param mixed $studyplanid Id of studyplan to export * @param string $format Export format [csv, json (default)] * @return array */ public static function export_plan($studyplanid, $format = "json") { try { // Validate permissions. webservicehelper::require_capabilities(self::CAP_EDIT, studyplan::find_by_id($studyplanid)->context()); $plan = studyplan::find_by_id($studyplanid); return $plan->export_plan(); } catch (\webservice_access_exception $x) { return [ "format" => "", "content" => ""]; } } /** * Parameter description for webservice function export_studylines */ public static function export_page_parameters() : \external_function_parameters { return new \external_function_parameters( [ "page_id" => new \external_value(PARAM_INT, 'id of plan to export '), "format" => new \external_value(PARAM_TEXT, 'export format', VALUE_OPTIONAL), ] ); } /** * Return value description for webservice function export_studylines */ public static function export_page_returns() : \external_description { return studyplanpage::export_structure(); } /** * Export studylines for a studyplan * @param mixed $studyplanid Id of the studyplan to export the studylines for * @param string $format Export format [csv, json (default)] * @return array */ public static function export_page($pageid,$format="json") { try { $page = studyplanpage::find_by_id($pageid); webservicehelper::require_capabilities(self::CAP_EDIT, $page->studyplan()->context()); if ($format == "csv") { return $page->export_page_csv(); } else { return $page->export_page(); } } catch (\webservice_access_exception $x) { return [ "format" => "", "content" => ""]; } } /**************************** * * * import studyplan * * * ****************************/ /** * Parameter description for webservice function import_plan */ public static function import_plan_parameters() : \external_function_parameters { return new \external_function_parameters( [ "content" => new \external_value(PARAM_RAW, 'import file content'), "format" => new \external_value(PARAM_TEXT, 'import format'), "context_id" => new \external_value(PARAM_INT, 'context of the study plan', VALUE_DEFAULT), ] ); } /** * Return value description for webservice function import_plan */ public static function import_plan_returns() : \external_description { return success::structure(); } /** * Import studyplan from file * @param string $content Content of file * @param string $format Format of file * @param int $contextid ID of context to import to * @return array Success/fail model */ public static function import_plan($content, $format = "application/json", $contextid = 1) { try { // Validate import context. $context = webservicehelper::find_context($contextid); webservicehelper::require_capabilities(self::CAP_EDIT, $context); $result = studyplan::import_studyplan($content, $format, $contextid); return (new success($result, "During study plan import"))->model(); } catch (\webservice_access_exception $x) { return success::fail("Access denied")->model(); } } /** * Parameter description for webservice function import_studylines */ public static function import_pages_parameters() : \external_function_parameters { return new \external_function_parameters( [ "studyplan_id" => new \external_value(PARAM_INT, 'id of studyplan to import to '), "content" => new \external_value(PARAM_RAW, 'import file content'), "format" => new \external_value(PARAM_TEXT, 'import format'), ] ); } /** * Return value description for webservice function import_studylines */ public static function import_pages_returns() : \external_description { return success::structure(); } /** * Import studylines into existing studtplan * @param int $studyplanid ID of studyplan to import to * @param string $content Content of file * @param string $format Format of file * @return array Success/fail model */ public static function import_pages($studyplanid, $content, $format = "application/json") { try { $plan = studyplan::find_by_id($studyplanid); // Validate import context. webservicehelper::require_capabilities(self::CAP_EDIT, $plan->context()); $result = $plan->import_pages($content, $format); return ($result ? success::success() : success::fail())->model(); } catch (\webservice_access_exception $x) { return success::fail("Access denied")->model(); } } /** * Parameter description for webservice function import_studylines */ public static function import_studylines_parameters() : \external_function_parameters { return new \external_function_parameters( [ "page_id" => new \external_value(PARAM_INT, 'id of studyplan page to import to '), "content" => new \external_value(PARAM_RAW, 'import file content'), "format" => new \external_value(PARAM_TEXT, 'import format'), ] ); } /** * Return value description for webservice function import_studylines */ public static function import_studylines_returns() : \external_description { return success::structure(); } /** * Import studylines into existing studtplan * @param int $studyplanid ID of studyplan to import to * @param string $content Content of file * @param string $format Format of file * @return array Success/fail model */ public static function import_studylines($page_id, $content, $format = "application/json") { try { $page = studyplanpage::find_by_id($page_id); // Validate import context. webservicehelper::require_capabilities(self::CAP_EDIT, $page->studyplan()->context()); $result = $page->import_studylines($content, $format); return ($result ? success::success() : success::fail())->model(); } catch (\webservice_access_exception $x) { return success::fail("Access denied")->model(); } } /******************************************************** * * * Read and write course module title and desc * * Used only in bistate aggregation method * * * ********************************************************/ /** * Parameter description for webservice function submit_cm_editform */ public static function submit_cm_editform_parameters() : \external_function_parameters { return new \external_function_parameters( [ "cmid" => new \external_value(PARAM_INT, 'id of course module'), "formdata" => new \external_value(PARAM_RAW, 'url encoded form data'), ] ); } /** * Return value description for webservice function submit_cm_editform */ public static function submit_cm_editform_returns() : \external_description { return success::structure(); } /** * Submit hacked course activity edit form to edit just name and description * @deprecated will remove hacked edit form in the future * @param mixed $cmid * @param mixed $formdata * @return array Success/fail structure */ public static function submit_cm_editform($cmid, $formdata) { global $CFG; global $DB; // Check the course module exists. $cm = \get_coursemodule_from_id('', $cmid, 0, false, MUST_EXIST); // Get some context ready. $context = \context_course::instance($cm->course); self::validate_context($context); // Check the course exists. $course = \get_course($cm->course); // Require_login. require_login($course, false, $cm); // Needed to setup proper $COURSE. // Get needed info to create the correct form. list($cm, $context, $module, $data, $cw) = \get_moduleinfo_data($cm, $course); $modmoodleform = "$CFG->dirroot/mod/$module->name/mod_form.php"; if (file_exists($modmoodleform)) { require_once($modmoodleform); } else { throw new \moodle_exception('noformdesc', 'error'); } $mformclassname = 'mod_'.$module->name.'_mod_form'; // Now hack the received data into $_POST, so the mform thinks it has been submitted "normally". foreach (explode("&", $formdata) as $pair) { $p = explode("=", $pair, 2); $k = urldecode($p[0]); $v = (count($p) > 1) ? urldecode($p[1]) : ""; if (strpos($k, "[") > 0 && strpos($k, "]") == strlen($k) - 1) { // Its a bracketet field, like filename[text] which should be separated and put into a named array. list($k, $h) = explode("[", $k, 2); if (strlen($k) > 0 && strlen($h) > 1) { $h = rtrim($h, "]"); if (!array_key_exists($k, $_POST) || !is_array($_POST[$k])) { $_POST[$k] = []; } $_POST[$k][$h] = $v; } } else { $_POST[$k] = $v; } } // Now create the mform and update the module... // Update_moduleinfo() actually needs the mform somehow. Hence the ugly hacks. $mform = new $mformclassname($data, $cw->section, $cm, $course); $mform->set_data($data); if ($fromform = $mform->get_data()) { update_moduleinfo($cm, $fromform, $course, $mform); return success::success()->model(); } else { return success::fail("MForm did not recognize submission data")->model(); } } /************************ * * * edit_period * * * ************************/ /** * Parameter description for webservice function get_period */ public static function get_period_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of study item'), ]); } /** * Return value description for webservice function get_period */ public static function get_period_returns() : \external_description { return period::structure(); } /** * Get period information * @param mixed $id Id of period to retrieve * @return array */ public static function get_period($id) { $p = period::find_by_id($id); // Public data - no rights check needed. \external_api::validate_context($p->page()->studyplan()->context()); return $p->model(); } /** * Parameter description for webservice function edit_period */ public static function edit_period_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of study item'), "fullname" => new \external_value(PARAM_TEXT, 'Full name of period'), "shortname" => new \external_value(PARAM_TEXT, 'Short name of period'), "startdate" => new \external_value(PARAM_TEXT, 'start date of period'), "enddate" => new \external_value(PARAM_TEXT, 'end date of period'), ]); } /** * Return value description for webservice function edit_period */ public static function edit_period_returns() : \external_description { return period::structure(); } /** * Edit period information * @param mixed $id * @param mixed $fullname * @param mixed $shortname * @param mixed $startdate * @param mixed $enddate * @return array */ public static function edit_period($id, $fullname, $shortname, $startdate, $enddate) { $p = period::find_by_id($id); webservicehelper::require_capabilities(self::CAP_EDIT, $p->page()->studyplan()->context()); $p->edit([ 'fullname' => $fullname, 'shortname' => $shortname, 'startdate' => $startdate, 'enddate' => $enddate, ]); return $p->model(); } /************************ * * * Change course timing * * * ************************/ /** * Parameter description for webservice function course_period_timing */ public static function course_period_timing_parameters() : \external_function_parameters { return new \external_function_parameters( [ "period_id" => new \external_value(PARAM_INT, 'Period number within page'), "course_id" => new \external_value(PARAM_INT, 'Id of course to adjust dates for'), "span" => new \external_value(PARAM_INT, 'Period span (default 1)', VALUE_DEFAULT), ]); } /** * Return value description for webservice function course_period_timing */ public static function course_period_timing_returns() : \external_description { return courseinfo::editor_structure(); } /** * Match course start/end to period start/end times * @param mixed $periodid * @param mixed $courseid * @param int $span * @return array */ public static function course_period_timing($periodid, $courseid, $span = 1) { global $DB; $period = period::find_by_id($periodid); $periodnr = $period->period(); $page = $period->page(); // Check for studyplan edit permissions. webservicehelper::require_capabilities(self::CAP_EDIT, $page->studyplan()->context()); $course = \get_course($courseid); $coursecontext = \context_course::instance($courseid); if (webservicehelper::has_capabilities("moodle/course:update", $coursecontext)) { // Get the proper list of all the periods for this page. $periods = period::find_for_page($page); $pstart = $periods[$periodnr]; // Determine end period number - Clip span between 1 and last period. if ($span <= 1) { $pend = $pstart; } else if ($periodnr + ($span - 1) > $page->periods()) { $pend = $periods[$page->periods()]; } else { $pend = $periods[$periodnr + ($span - 1)]; } // First disable the automatic end date option, since it messes with the timing. // Unfortunately there is still no default option to turn this off . $record = $DB->get_record("course_format_options", ["courseid" => $course->id, "name" => "automaticenddate"]); if ($record && $record->value) { $record->value = false; $DB->update_record("course_format_options", $record); } // Actually perform the timing changes, while also updating the module times. // Like what happens on a course "reset". reset_course_userdata((object)[ 'id' => $course->id, 'reset_start_date' => $pstart->startdate()->getTimestamp(), 'reset_end_date' => $pend->enddate()->getTimestamp(), 'reset_start_date_old' => $course->startdate, 'reset_end_date_old' => $course->enddate, ]); // Purge course cache so the dates are properly reflected. \course_modinfo::purge_course_cache($course->id); return (new courseinfo($course->id))->editor_model(); } else { // Probably should return a nice message. throw new \webservice_access_exception("You do not have date change permissions on this course"); } } /** * Parameter description for webservice function set_studyitem_span */ public static function set_studyitem_span_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of study item'), "span" => new \external_value(PARAM_INT, 'span of item'), ]); } /** * Return value description for webservice function set_studyitem_span */ public static function set_studyitem_span_returns() : \external_description { return studyitem::editor_structure(); } /** * Set studyitem span to one or more periods * @param mixed $id * @param null $span * @return array */ public static function set_studyitem_span($id, $span = null) { $o = studyitem::find_by_id($id); webservicehelper::require_capabilities(self::CAP_EDIT, $o->context()); $config = [ 'span' => $span]; $o->edit($config); return $o->editor_model(); } /************************ * * * bulk_course_timing * * * ************************/ /** * Parameter description for webservice function delete_studyplan */ public static function bulk_course_timing_parameters() : \external_function_parameters { return new \external_function_parameters( [ "page_id" => new \external_value(PARAM_INT, 'id of studyplanpage'), ] ); } /** * Return value description for webservice function delete_studyplan */ public static function bulk_course_timing_returns() : \external_description { return success::structure(); } /** * Delete a studyplan * @param mixed $id Id of the studyplan * @param bool $force Force deletion, even though studyplan is not empty * @return array Succes/fail model */ public static function bulk_course_timing($pageid) { $page = studyplanpage::find_by_id($pageid); // Validate if the requesting user has the right to edit the plan in it's current context. webservicehelper::require_capabilities(self::CAP_EDIT, $page->studyplan()->context()); $result = true; $message = []; $periods = period::find_for_page($page); $studylines = studyline::find_page_children($page); foreach ($studylines as $line) { $studyitems = studyitem::find_studyline_children($line); foreach ($studyitems as $item) { if ($item->type() == studyitem::COURSE) { if ( $item->slot() <= $page->periods() ) { $period = $periods[$item->slot()]; try { self::course_period_timing($period->id(),$item->courseid(),$item->span()); } catch (\webservice_access_exception $x) { $result &= false; $course = \get_course($item->courseid()); $message[] = "Course {$course->shortname}: {$x->debuginfo}"; } catch (\Exception $x) { $result &= false; $course = \get_course($item->courseid()); $xclass = \get_class($x); $message[] = "Course {$course->shortname}: {$xclass} {$x->getMessage()}"; } } } } } return (new success($result,implode("
\n",$message)))->model(); } /************************ * * * delete_studyplanpage * * * ************************/ /** * Parameter description for webservice function delete_studyplan */ public static function delete_studyplanpage_parameters() : \external_function_parameters { return new \external_function_parameters( [ "id" => new \external_value(PARAM_INT, 'id of studyplanpage'), "force" => new \external_value(PARAM_BOOL, 'force deletion', VALUE_DEFAULT), ] ); } /** * Return value description for webservice function delete_studyplan */ public static function delete_studyplanpage_returns() : \external_description { return success::structure(); } /** * Delete a studyplan * @param mixed $id Id of the studyplan * @param bool $force Force deletion, even though studyplan is not empty * @return array Succes/fail model */ public static function delete_studyplanpage($id, $force = false) { $o = studyplanpage::find_by_id($id); $p = $o->studyplan(); // Validate if the requesting user has the right to edit the plan in it's current context. webservicehelper::require_capabilities(self::CAP_EDIT, $p->context()); return $o->delete(!!$force)->model(); } }