mirror of
https://github.com/moodle/moodle.git
synced 2025-08-08 02:16:41 +02:00
MDL-71165 course: core_course_update_course external method
This commit is contained in:
parent
a9b0f4dafe
commit
6347b916bc
12 changed files with 2052 additions and 1 deletions
152
course/classes/external/update_course.php
vendored
Normal file
152
course/classes/external/update_course.php
vendored
Normal file
|
@ -0,0 +1,152 @@
|
|||
<?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/>.
|
||||
|
||||
namespace core_course\external;
|
||||
|
||||
defined('MOODLE_INTERNAL') || die();
|
||||
|
||||
global $CFG;
|
||||
require_once($CFG->libdir . '/externallib.php');
|
||||
|
||||
use external_api;
|
||||
use external_function_parameters;
|
||||
use external_value;
|
||||
use external_multiple_structure;
|
||||
use moodle_exception;
|
||||
use coding_exception;
|
||||
use context_course;
|
||||
|
||||
/**
|
||||
* External secrvie to update the course from the course editor components.
|
||||
*
|
||||
* @package core_course
|
||||
* @copyright 2021 Ferran Recio <moodle@moodle.com>
|
||||
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
||||
* @since Moodle 4.0
|
||||
*/
|
||||
class update_course extends external_api {
|
||||
|
||||
/**
|
||||
* Webservice parameters.
|
||||
*
|
||||
* @return external_function_parameters
|
||||
*/
|
||||
public static function execute_parameters(): external_function_parameters {
|
||||
return new external_function_parameters(
|
||||
[
|
||||
'action' => new external_value(
|
||||
PARAM_ALPHANUMEXT,
|
||||
'action: cm_hide, cm_show, section_hide, section_show, cm_moveleft...',
|
||||
VALUE_REQUIRED
|
||||
),
|
||||
'courseid' => new external_value(PARAM_INT, 'course id', VALUE_REQUIRED),
|
||||
'ids' => new external_multiple_structure(
|
||||
new external_value(PARAM_INT, 'Target id'),
|
||||
'Affected ids',
|
||||
VALUE_DEFAULT,
|
||||
[]
|
||||
),
|
||||
'targetsectionid' => new external_value(
|
||||
PARAM_INT, 'Optional target section id', VALUE_DEFAULT, null
|
||||
),
|
||||
'targetcmid' => new external_value(
|
||||
PARAM_INT, 'Optional target cm id', VALUE_DEFAULT, null
|
||||
),
|
||||
]
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* This webservice will execute any action from the course editor. The default actions
|
||||
* are located in core_course\stateactions but the format plugin can extend that class
|
||||
* in format_XXX\course.
|
||||
*
|
||||
* The specific action methods will register in a core_course\stateupdates all the affected
|
||||
* sections, cms and course attribute. This object (in JSON) will be send back to the
|
||||
* frontend editor to refresh the updated state elements.
|
||||
*
|
||||
* By default, core_course\stateupdates will register only create, delete and update events
|
||||
* on cms, sections and the general course data. However, if some plugin needs adhoc messages for
|
||||
* its own mutation module, it extend this class in format_XXX\course.
|
||||
*
|
||||
* @param string $action the action name to execute
|
||||
* @param int $courseid the course id
|
||||
* @param int[] $ids the affected ids (section or cm depending on the action)
|
||||
* @param int $targetsectionid optional target section id (for move action)
|
||||
* @param int $targetcmid optional target cm id (for move action)
|
||||
* @return string Course state in JSON
|
||||
*/
|
||||
public static function execute(string $action, int $courseid, array $ids = [],
|
||||
?int $targetsectionid = null, ?int $targetcmid = null): string {
|
||||
global $CFG;
|
||||
|
||||
require_once($CFG->dirroot . '/course/lib.php');
|
||||
|
||||
$params = external_api::validate_parameters(self::execute_parameters(), [
|
||||
'action' => $action,
|
||||
'courseid' => $courseid,
|
||||
'ids' => $ids,
|
||||
'targetsectionid' => $targetsectionid,
|
||||
'targetcmid' => $targetcmid,
|
||||
]);
|
||||
$action = $params['action'];
|
||||
$courseid = $params['courseid'];
|
||||
$ids = $params['ids'];
|
||||
$targetsectionid = $params['targetsectionid'];
|
||||
$targetcmid = $params['targetcmid'];
|
||||
|
||||
self::validate_context(context_course::instance($courseid));
|
||||
|
||||
$courseformat = course_get_format($courseid);
|
||||
|
||||
// Create a course changes tracker object.
|
||||
$defaultupdatesclass = 'core_course\\stateupdates';
|
||||
$updatesclass = 'format_' . $courseformat->get_format() . '\\stateupdates';
|
||||
if (!class_exists($updatesclass)) {
|
||||
$updatesclass = $defaultupdatesclass;
|
||||
}
|
||||
$updates = new $updatesclass($courseformat);
|
||||
|
||||
if (!is_a($updates, $defaultupdatesclass)) {
|
||||
throw new coding_exception("The \"$updatesclass\" class must extend \"$defaultupdatesclass\"");
|
||||
}
|
||||
|
||||
// Get the actions class from the course format.
|
||||
$actionsclass = 'format_'. $courseformat->get_format().'\\stateactions';
|
||||
if (!class_exists($actionsclass)) {
|
||||
$actionsclass = 'core_course\\stateactions';
|
||||
}
|
||||
$actions = new $actionsclass();
|
||||
|
||||
if (!is_callable([$actions, $action])) {
|
||||
throw new moodle_exception("Invalid course state action $action in ".get_class($actions));
|
||||
}
|
||||
|
||||
// Execute the action.
|
||||
$actions->$action($updates, $courseformat->get_course(), $ids, $targetsectionid, $targetcmid);
|
||||
|
||||
return json_encode($updates);
|
||||
}
|
||||
|
||||
/**
|
||||
* Webservice returns.
|
||||
*
|
||||
* @return external_value
|
||||
*/
|
||||
public static function execute_returns(): external_value {
|
||||
return new external_value(PARAM_RAW, 'Encoded course update JSON');
|
||||
}
|
||||
}
|
237
course/classes/stateactions.php
Normal file
237
course/classes/stateactions.php
Normal file
|
@ -0,0 +1,237 @@
|
|||
<?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/>.
|
||||
|
||||
namespace core_course;
|
||||
|
||||
use core_course\stateupdates;
|
||||
use stdClass;
|
||||
use course_modinfo;
|
||||
use moodle_exception;
|
||||
|
||||
/**
|
||||
* Contains the core course state actions.
|
||||
*
|
||||
* The methods from this class should be executed via "core_course_edit" web service.
|
||||
*
|
||||
* Each format plugin could extend this class to provide new actions to the editor.
|
||||
* Extended classes should be locate in "format_XXX\course" namespace and
|
||||
* extends core_course\stateactions.
|
||||
*
|
||||
* @package core_course
|
||||
* @copyright 2021 Ferran Recio <ferran@moodle.com>
|
||||
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
||||
*/
|
||||
class stateactions {
|
||||
|
||||
/**
|
||||
* Add the update messages of the updated version of any cm and section related to the cm ids.
|
||||
*
|
||||
* This action is mainly used by legacy actions to partially update the course state when the
|
||||
* result of core_course_edit_module is not enough to generate the correct state data.
|
||||
*
|
||||
* @param stateupdates $updates the affected course elements track
|
||||
* @param stdClass $course the course object
|
||||
* @param int[] $ids the list of affected course module ids
|
||||
* @param int $targetsectionid optional target section id
|
||||
* @param int $targetcmid optional target cm id
|
||||
*/
|
||||
public function cm_state(
|
||||
stateupdates $updates,
|
||||
stdClass $course,
|
||||
array $ids,
|
||||
?int $targetsectionid = null,
|
||||
?int $targetcmid = null
|
||||
): void {
|
||||
|
||||
// Collect all section and cm to return.
|
||||
$cmids = [];
|
||||
foreach ($ids as $cmid) {
|
||||
$cmids[$cmid] = true;
|
||||
}
|
||||
if ($targetcmid) {
|
||||
$cmids[$targetcmid] = true;
|
||||
}
|
||||
|
||||
$sectionids = [];
|
||||
if ($targetsectionid) {
|
||||
$this->validate_sections($course, [$targetsectionid], __FUNCTION__);
|
||||
$sectionids[$targetsectionid] = true;
|
||||
}
|
||||
|
||||
$this->validate_cms($course, array_keys($cmids), __FUNCTION__);
|
||||
|
||||
$modinfo = course_modinfo::instance($course);
|
||||
|
||||
foreach (array_keys($cmids) as $cmid) {
|
||||
|
||||
// Add this action to updates array.
|
||||
$updates->add_cm_update($cmid);
|
||||
|
||||
$cm = $modinfo->get_cm($cmid);
|
||||
$sectionids[$cm->section] = true;
|
||||
}
|
||||
|
||||
foreach (array_keys($sectionids) as $sectionid) {
|
||||
$updates->add_section_update($sectionid);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Add the update messages of the updated version of any cm and section related to the section ids.
|
||||
*
|
||||
* This action is mainly used by legacy actions to partially update the course state when the
|
||||
* result of core_course_edit_module is not enough to generate the correct state data.
|
||||
*
|
||||
* @param stateupdates $updates the affected course elements track
|
||||
* @param stdClass $course the course object
|
||||
* @param int[] $ids the list of affected course section ids
|
||||
* @param int $targetsectionid optional target section id
|
||||
* @param int $targetcmid optional target cm id
|
||||
*/
|
||||
public function section_state(
|
||||
stateupdates $updates,
|
||||
stdClass $course,
|
||||
array $ids,
|
||||
?int $targetsectionid = null,
|
||||
?int $targetcmid = null
|
||||
): void {
|
||||
|
||||
$cmids = [];
|
||||
if ($targetcmid) {
|
||||
$this->validate_cms($course, [$targetcmid], __FUNCTION__);
|
||||
$cmids[$targetcmid] = true;
|
||||
}
|
||||
|
||||
$sectionids = [];
|
||||
foreach ($ids as $sectionid) {
|
||||
$sectionids[$sectionid] = true;
|
||||
}
|
||||
if ($targetsectionid) {
|
||||
$sectionids[$targetsectionid] = true;
|
||||
}
|
||||
|
||||
$this->validate_sections($course, array_keys($sectionids), __FUNCTION__);
|
||||
|
||||
$modinfo = course_modinfo::instance($course);
|
||||
|
||||
foreach (array_keys($sectionids) as $sectionid) {
|
||||
$sectioninfo = $modinfo->get_section_info_by_id($sectionid);
|
||||
$updates->add_section_update($sectionid);
|
||||
// Add cms.
|
||||
if (empty($modinfo->sections[$sectioninfo->section])) {
|
||||
continue;
|
||||
}
|
||||
|
||||
foreach ($modinfo->sections[$sectioninfo->section] as $modnumber) {
|
||||
$mod = $modinfo->cms[$modnumber];
|
||||
if ($mod->is_visible_on_course_page()) {
|
||||
$cmids[$mod->id] = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
foreach (array_keys($cmids) as $cmid) {
|
||||
// Add this action to updates array.
|
||||
$updates->add_cm_update($cmid);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Add all the update messages from the complete course state.
|
||||
*
|
||||
* This action is mainly used by legacy actions to partially update the course state when the
|
||||
* result of core_course_edit_module is not enough to generate the correct state data.
|
||||
*
|
||||
* @param stateupdates $updates the affected course elements track
|
||||
* @param stdClass $course the course object
|
||||
* @param int[] $ids the list of affected course module ids (not used)
|
||||
* @param int $targetsectionid optional target section id (not used)
|
||||
* @param int $targetcmid optional target cm id (not used)
|
||||
*/
|
||||
public function course_state(
|
||||
stateupdates $updates,
|
||||
stdClass $course,
|
||||
array $ids = [],
|
||||
?int $targetsectionid = null,
|
||||
?int $targetcmid = null
|
||||
): void {
|
||||
|
||||
$modinfo = course_modinfo::instance($course);
|
||||
|
||||
$updates->add_course_update();
|
||||
|
||||
// Add sections updates.
|
||||
$sections = $modinfo->get_section_info_all();
|
||||
$sectionids = [];
|
||||
foreach ($sections as $sectioninfo) {
|
||||
$sectionids[] = $sectioninfo->id;
|
||||
}
|
||||
if (!empty($sectionids)) {
|
||||
$this->section_state($updates, $course, $sectionids);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks related to sections: course format support them, all given sections exist and topic 0 is not included.
|
||||
*
|
||||
* @param stdClass $course The course where given $sectionids belong.
|
||||
* @param array $sectionids List of sections to validate.
|
||||
* @param string|null $info additional information in case of error (default null).
|
||||
* @throws moodle_exception if any id is not valid
|
||||
*/
|
||||
protected function validate_sections(stdClass $course, array $sectionids, ?string $info = null): void {
|
||||
global $DB;
|
||||
|
||||
if (empty($sectionids)) {
|
||||
throw new moodle_exception('emptysectionids', 'core', null, $info);
|
||||
}
|
||||
|
||||
// No section actions are allowed if course format does not support sections.
|
||||
$courseformat = course_get_format($course->id);
|
||||
if (!$courseformat->uses_sections()) {
|
||||
throw new moodle_exception('sectionactionnotsupported', 'core', null, $info);
|
||||
}
|
||||
|
||||
list($insql, $inparams) = $DB->get_in_or_equal($sectionids, SQL_PARAMS_NAMED);
|
||||
|
||||
// Check if all the given sections exist.
|
||||
$couintsections = $DB->count_records_select('course_sections', "id $insql", $inparams);
|
||||
if ($couintsections != count($sectionids)) {
|
||||
throw new moodle_exception('unexistingsectionid', 'core', null, $info);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks related to course modules: all given cm exist.
|
||||
*
|
||||
* @param stdClass $course The course where given $cmids belong.
|
||||
* @param array $cmids List of course module ids to validate.
|
||||
* @param string $info additional information in case of error.
|
||||
* @throws moodle_exception if any id is not valid
|
||||
*/
|
||||
protected function validate_cms(stdClass $course, array $cmids, ?string $info = null): void {
|
||||
|
||||
if (empty($cmids)) {
|
||||
throw new moodle_exception('emptycmids', 'core', null, $info);
|
||||
}
|
||||
|
||||
$moduleinfo = get_fast_modinfo($course->id);
|
||||
$intersect = array_intersect($cmids, array_keys($moduleinfo->get_cms()));
|
||||
if (count($cmids) != count($intersect)) {
|
||||
throw new moodle_exception('unexistingcmid', 'core', null, $info);
|
||||
}
|
||||
}
|
||||
}
|
200
course/classes/stateupdates.php
Normal file
200
course/classes/stateupdates.php
Normal file
|
@ -0,0 +1,200 @@
|
|||
<?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/>.
|
||||
|
||||
namespace core_course;
|
||||
|
||||
use coding_exception;
|
||||
use core_course\course_format;
|
||||
use renderer_base;
|
||||
use stdClass;
|
||||
use course_modinfo;
|
||||
use JsonSerializable;
|
||||
|
||||
/**
|
||||
* Class to track state actions.
|
||||
*
|
||||
* The methods from this class should be executed via "stateactions" methods.
|
||||
*
|
||||
* Each format plugin could extend this class to provide new updates to the frontend
|
||||
* mutation module.
|
||||
* Extended classes should be locate in "format_XXX\course" namespace and
|
||||
* extends core_course\stateupdates.
|
||||
*
|
||||
* @package core_course
|
||||
* @copyright 2021 Ferran Recio <ferran@moodle.com>
|
||||
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
||||
*/
|
||||
class stateupdates implements JsonSerializable {
|
||||
|
||||
/** @var course_format format the course format */
|
||||
protected $format;
|
||||
|
||||
/** @var renderer_base renderer format renderer */
|
||||
protected $output;
|
||||
|
||||
/** @var array the tracked updates */
|
||||
protected $updates;
|
||||
|
||||
/**
|
||||
* State update class constructor.
|
||||
*
|
||||
* @param course_format $format Course format.
|
||||
*/
|
||||
public function __construct(course_format $format) {
|
||||
global $PAGE;
|
||||
|
||||
$this->format = $format;
|
||||
$this->output = $this->format->get_renderer($PAGE);
|
||||
$this->updates = [];
|
||||
}
|
||||
|
||||
/**
|
||||
* Return the data to serialize the current track in JSON.
|
||||
*
|
||||
* @return stdClass the statement data structure
|
||||
*/
|
||||
public function jsonSerialize(): array {
|
||||
return $this->updates;
|
||||
}
|
||||
|
||||
/**
|
||||
* Add track about a general course state change.
|
||||
*/
|
||||
public function add_course_update(): void {
|
||||
$courseclass = $this->format->get_output_classname('course_format\state');
|
||||
$currentstate = new $courseclass($this->format);
|
||||
$this->add_update('course', 'update', $currentstate->export_for_template($this->output));
|
||||
}
|
||||
|
||||
/**
|
||||
* Add track about a section state update.
|
||||
*
|
||||
* @param int $sectionid The affected section id.
|
||||
*/
|
||||
public function add_section_update(int $sectionid): void {
|
||||
$this->create_or_update_section($sectionid, 'update');
|
||||
}
|
||||
|
||||
/**
|
||||
* Add track about a new section created.
|
||||
*
|
||||
* @param int $sectionid The affected section id.
|
||||
*/
|
||||
public function add_section_create(int $sectionid): void {
|
||||
$this->create_or_update_section($sectionid, 'create');
|
||||
}
|
||||
|
||||
/**
|
||||
* Add track about section created or updated.
|
||||
*
|
||||
* @param int $sectionid The affected section id.
|
||||
* @param string $action The action to track for the section ('create' or 'update).
|
||||
*/
|
||||
protected function create_or_update_section(int $sectionid, string $action): void {
|
||||
if ($action != 'create' && $action != 'update') {
|
||||
throw new coding_exception(
|
||||
"Invalid action passed ($action) to create_or_update_section. Only 'create' and 'update' are valid."
|
||||
);
|
||||
}
|
||||
$course = $this->format->get_course();
|
||||
$modinfo = course_modinfo::instance($course);
|
||||
|
||||
$section = $modinfo->get_section_info_by_id($sectionid, MUST_EXIST);
|
||||
|
||||
if (!$section->uservisible) {
|
||||
return;
|
||||
}
|
||||
|
||||
$sectionclass = $this->format->get_output_classname('section_format\state');
|
||||
$currentstate = new $sectionclass($this->format, $section);
|
||||
|
||||
$this->add_update('section', $action, $currentstate->export_for_template($this->output));
|
||||
}
|
||||
|
||||
/**
|
||||
* Add track about a section deleted.
|
||||
*
|
||||
* @param int $sectionid The affected section id.
|
||||
*/
|
||||
public function add_section_delete(int $sectionid): void {
|
||||
$this->add_update('section', 'delete', (object)['id' => $sectionid]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Add track about a course module state update.
|
||||
*
|
||||
* @param int $cmid the affected course module id
|
||||
*/
|
||||
public function add_cm_update(int $cmid): void {
|
||||
$this->create_or_update_cm($cmid, 'update');
|
||||
}
|
||||
|
||||
/**
|
||||
* Add track about a course module created.
|
||||
*
|
||||
* @param int $cmid the affected course module id
|
||||
*/
|
||||
public function add_cm_create(int $cmid): void {
|
||||
$this->create_or_update_cm($cmid, 'create', true);
|
||||
}
|
||||
|
||||
/**
|
||||
* Add track about section created or updated.
|
||||
*
|
||||
* @param int $cmid The affected course module id.
|
||||
* @param string $action The action to track for the section ('create' or 'update').
|
||||
*/
|
||||
protected function create_or_update_cm(int $cmid, string $action): void {
|
||||
$modinfo = course_modinfo::instance($this->format->get_course());
|
||||
|
||||
$cm = $modinfo->get_cm($cmid);
|
||||
$section = $modinfo->get_section_info_by_id($cm->section);
|
||||
|
||||
if (!$section->uservisible || !$cm->is_visible_on_course_page()) {
|
||||
return;
|
||||
}
|
||||
|
||||
$cmclass = $this->format->get_output_classname('cm_format\state');
|
||||
$currentstate = new $cmclass($this->format, $section, $cm);
|
||||
|
||||
$this->add_update('cm', $action, $currentstate->export_for_template($this->output));
|
||||
}
|
||||
|
||||
/**
|
||||
* Add track about a course module deleted.
|
||||
*
|
||||
* @param int $cmid the affected course module id
|
||||
*/
|
||||
public function add_cm_delete(int $cmid): void {
|
||||
$this->add_update('cm', 'delete', (object)['id' => $cmid]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Add a valid update message to the update list.
|
||||
*
|
||||
* @param string $name: the update name
|
||||
* @param string $action: the update action (usually update, create, delete)
|
||||
* @param stdClass $fields: the object fields
|
||||
*/
|
||||
protected function add_update(string $name, string $action, stdClass $fields): void {
|
||||
$this->updates[] = (object)[
|
||||
'name' => $name,
|
||||
'action' => $action,
|
||||
'fields' => $fields,
|
||||
];
|
||||
}
|
||||
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue