mirror of
https://gh.llkk.cc/https://github.com/h5p/moodle-mod_hvp.git
synced 2026-03-04 13:36:41 +08:00
506 lines
16 KiB
PHP
506 lines
16 KiB
PHP
<?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/>.
|
|
/**
|
|
* Responsible for handling AJAX requests related to H5P.
|
|
*
|
|
* @package mod_hvp
|
|
* @copyright 2016 Joubel AS <contact@joubel.com>
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
*/
|
|
|
|
use mod_hvp\framework;
|
|
|
|
define('AJAX_SCRIPT', true);
|
|
require(__DIR__ . '/../../config.php');
|
|
require_once($CFG->libdir . '/filelib.php');
|
|
require_once("locallib.php");
|
|
|
|
require_login();
|
|
|
|
$action = required_param('action', PARAM_ALPHA);
|
|
switch($action) {
|
|
|
|
/*
|
|
* Handle user data reporting
|
|
*
|
|
* Type: HTTP POST
|
|
*
|
|
* Parameters:
|
|
* - content_id
|
|
* - data_type
|
|
* - sub_content_id
|
|
*/
|
|
case 'contentsuserdata':
|
|
\mod_hvp\content_user_data::handle_ajax();
|
|
break;
|
|
|
|
/*
|
|
* Handle restricting H5P libraries
|
|
*
|
|
* Type: HTTP GET
|
|
*
|
|
* Parameters:
|
|
* - library_id
|
|
* - restrict (0 or 1)
|
|
* - token
|
|
*/
|
|
case 'restrictlibrary':
|
|
|
|
// Check permissions.
|
|
$context = \context_system::instance();
|
|
if (!has_capability('mod/hvp:restrictlibraries', $context)) {
|
|
\H5PCore::ajaxError(get_string('nopermissiontorestrict', 'hvp'));
|
|
http_response_code(403);
|
|
break;
|
|
}
|
|
|
|
$libraryid = required_param('library_id', PARAM_INT);
|
|
$restrict = required_param('restrict', PARAM_INT);
|
|
|
|
if (!\H5PCore::validToken('library_' . $libraryid, required_param('token', PARAM_RAW))) {
|
|
\H5PCore::ajaxError(get_string('invalidtoken', 'hvp'));
|
|
exit;
|
|
}
|
|
|
|
hvp_restrict_library($libraryid, $restrict);
|
|
header('Cache-Control: no-cache');
|
|
header('Content-Type: application/json');
|
|
echo json_encode(array(
|
|
'url' => (new moodle_url('/mod/hvp/ajax.php', array(
|
|
'action' => 'restrict_library',
|
|
'token' => \H5PCore::createToken('library_' . $libraryid),
|
|
'restrict' => ($restrict === '1' ? 0 : 1),
|
|
'library_id' => $libraryid
|
|
)))->out(false)));
|
|
break;
|
|
|
|
/*
|
|
* Collecting data needed by H5P content upgrade
|
|
*
|
|
* Type: HTTP GET
|
|
*
|
|
* Parameters:
|
|
* - library (Format: /<machine-name>/<major-version>/<minor-version>)
|
|
*/
|
|
case 'getlibrarydataforupgrade':
|
|
|
|
// Check permissions.
|
|
$context = \context_system::instance();
|
|
if (!has_capability('mod/hvp:updatelibraries', $context)) {
|
|
\H5PCore::ajaxError(get_string('nopermissiontoupgrade', 'hvp'));
|
|
http_response_code(403);
|
|
break;
|
|
}
|
|
|
|
$library = required_param('library', PARAM_TEXT);
|
|
$library = explode('/', substr($library, 1));
|
|
|
|
if (count($library) !== 3) {
|
|
http_response_code(422);
|
|
return;
|
|
}
|
|
|
|
$library = hvp_get_library_upgrade_info($library[0], $library[1], $library[2]);
|
|
|
|
header('Cache-Control: no-cache');
|
|
header('Content-Type: application/json');
|
|
print json_encode($library);
|
|
|
|
break;
|
|
|
|
/*
|
|
* Saving upgraded content, and returning next batch to process
|
|
*
|
|
* Type: HTTP POST
|
|
*
|
|
* Parameters:
|
|
* - library_id
|
|
*/
|
|
case 'libraryupgradeprogress':
|
|
// Check upgrade permissions.
|
|
$context = \context_system::instance();
|
|
if (!has_capability('mod/hvp:updatelibraries', $context)) {
|
|
\H5PCore::ajaxError(get_string('nopermissiontoupgrade', 'hvp'));
|
|
http_response_code(403);
|
|
break;
|
|
}
|
|
|
|
// Because of a confirmed bug in PHP, filter_input(INPUT_SERVER, ...)
|
|
// will return null on some versions of FCGI/PHP (5.4 and probably
|
|
// older versions as well), ref. https://bugs.php.net/bug.php?id=49184.
|
|
if ($_SERVER['REQUEST_METHOD'] === 'POST') {
|
|
$libraryid = required_param('library_id', PARAM_INT);
|
|
$out = hvp_content_upgrade_progress($libraryid);
|
|
header('Cache-Control: no-cache');
|
|
header('Content-Type: application/json');
|
|
print json_encode($out);
|
|
} else {
|
|
// Only allow POST.
|
|
http_response_code(405);
|
|
}
|
|
break;
|
|
|
|
/*
|
|
* Handle set finished / storing grades
|
|
*
|
|
* Type: HTTP GET
|
|
*
|
|
* Parameters:
|
|
* - contentId
|
|
* - score
|
|
* - maxScore
|
|
*/
|
|
case 'setfinished':
|
|
\mod_hvp\user_grades::handle_ajax();
|
|
break;
|
|
|
|
/*
|
|
* Saves a dynamically graded grade to the gradebook
|
|
*
|
|
* Type: HTTP POST
|
|
*
|
|
* Parameters:
|
|
* - subcontent_id
|
|
* - score
|
|
*/
|
|
case 'updatesubcontentscore':
|
|
\mod_hvp\user_grades::handle_dynamic_grading();
|
|
break;
|
|
|
|
/*
|
|
* Returns a grade
|
|
*
|
|
* Type: HTTP GET
|
|
*
|
|
* Parameters:
|
|
* - subcontent_id
|
|
*/
|
|
case 'getsubcontentscore':
|
|
\mod_hvp\user_grades::return_subcontent_grade();
|
|
break;
|
|
|
|
/*
|
|
* Provide data for results view
|
|
*
|
|
* Type: HTTP GET
|
|
*
|
|
* Parameters:
|
|
* int content_id
|
|
* int offset
|
|
* int limit
|
|
* int sortBy
|
|
* int sortDir
|
|
* string[] filters
|
|
*/
|
|
case 'results':
|
|
$results = new \mod_hvp\results();
|
|
$results->print_results();
|
|
break;
|
|
|
|
/*
|
|
* Load list of libraries or details for library.
|
|
*
|
|
* Parameters:
|
|
* string machineName
|
|
* int majorVersion
|
|
* int minorVersion
|
|
*/
|
|
case 'libraries':
|
|
if (!framework::has_editor_access('nopermissiontoviewcontenttypes')) {
|
|
break;
|
|
}
|
|
|
|
// Get parameters.
|
|
$name = optional_param('machineName', '', PARAM_TEXT);
|
|
$major = optional_param('majorVersion', 0, PARAM_INT);
|
|
$minor = optional_param('minorVersion', 0, PARAM_INT);
|
|
$editor = framework::instance('editor');
|
|
$language = optional_param('default-language', null, PARAM_RAW);
|
|
|
|
if (!empty($name)) {
|
|
$editor->ajax->action(H5PEditorEndpoints::SINGLE_LIBRARY, $name,
|
|
$major, $minor, framework::get_language(), '', '', $language);
|
|
|
|
new \mod_hvp\event(
|
|
'library', null,
|
|
null, null,
|
|
$name, $major . '.' . $minor
|
|
);
|
|
} else {
|
|
$editor->ajax->action(H5PEditorEndpoints::LIBRARIES);
|
|
}
|
|
|
|
break;
|
|
|
|
/*
|
|
* Load content type cache list to display available libraries in hub
|
|
*/
|
|
case 'contenttypecache':
|
|
if (!framework::has_editor_access('nopermissiontoviewcontenttypes')) {
|
|
break;
|
|
}
|
|
|
|
$editor = framework::instance('editor');
|
|
$editor->ajax->action(H5PEditorEndpoints::CONTENT_TYPE_CACHE);
|
|
break;
|
|
|
|
/*
|
|
* Handle file upload through the editor.
|
|
*
|
|
* Parameters:
|
|
* int contentId
|
|
* int contextId
|
|
*/
|
|
case 'files':
|
|
$token = required_param('token', PARAM_RAW);
|
|
$contentid = required_param('contentId', PARAM_INT);
|
|
if (!framework::has_editor_access('nopermissiontouploadfiles')) {
|
|
break;
|
|
}
|
|
|
|
$editor = framework::instance('editor');
|
|
$editor->ajax->action(H5PEditorEndpoints::FILES, $token, $contentid);
|
|
break;
|
|
|
|
/*
|
|
* Handle file upload through the editor.
|
|
*
|
|
* Parameters:
|
|
* raw token
|
|
* raw contentTypeUrl
|
|
*/
|
|
case 'libraryinstall':
|
|
$token = required_param('token', PARAM_RAW);
|
|
$machinename = required_param('id', PARAM_TEXT);
|
|
$editor = framework::instance('editor');
|
|
$editor->ajax->action(H5PEditorEndpoints::LIBRARY_INSTALL, $token, $machinename);
|
|
break;
|
|
|
|
/*
|
|
* Install libraries from h5p and retrieve content json
|
|
*
|
|
* Parameters:
|
|
* file h5p
|
|
*/
|
|
case 'libraryupload':
|
|
$token = required_param('token', PARAM_RAW);
|
|
if (!framework::has_editor_access('nopermissiontouploadcontent')) {
|
|
break;
|
|
}
|
|
|
|
$editor = framework::instance('editor');
|
|
$uploadpath = $_FILES['h5p']['tmp_name'];
|
|
$contentid = optional_param('contentId', 0, PARAM_INT);
|
|
$editor->ajax->action(H5PEditorEndpoints::LIBRARY_UPLOAD, $token, $uploadpath, $contentid);
|
|
break;
|
|
|
|
/*
|
|
* Record xAPI result from view
|
|
*/
|
|
case 'xapiresult':
|
|
\mod_hvp\xapi_result::handle_ajax();
|
|
break;
|
|
|
|
case 'translations':
|
|
if (!framework::has_editor_access('nopermissiontogettranslations')) {
|
|
break;
|
|
}
|
|
$language = required_param('language', PARAM_RAW);
|
|
$editor = framework::instance('editor');
|
|
$editor->ajax->action(H5PEditorEndpoints::TRANSLATIONS, $language);
|
|
break;
|
|
|
|
/*
|
|
* Handle filtering of parameters through AJAX.
|
|
*/
|
|
case 'filter':
|
|
$token = required_param('token', PARAM_RAW);
|
|
$libraryparameters = required_param('libraryParameters', PARAM_RAW);
|
|
|
|
if (!framework::has_editor_access('nopermissiontouploadfiles')) {
|
|
break;
|
|
}
|
|
|
|
$editor = framework::instance('editor');
|
|
$editor->ajax->action(H5PEditorEndpoints::FILTER, $token, $libraryparameters);
|
|
break;
|
|
|
|
/*
|
|
* Handle filtering of parameters through AJAX.
|
|
*/
|
|
case 'contenthubmetadatacache':
|
|
if (!framework::has_editor_access('nopermissiontoviewcontenthubcache')) {
|
|
break;
|
|
}
|
|
$editor = framework::instance('editor');
|
|
$editor->ajax->action(H5PEditorEndpoints::CONTENT_HUB_METADATA_CACHE, framework::get_language());
|
|
break;
|
|
|
|
case 'contenthubregistration':
|
|
// Check permission.
|
|
$context = \context_system::instance();
|
|
if (!has_capability('mod/hvp:contenthubregistration', $context)) {
|
|
\H5PCore::ajaxError(get_string('contenthub:nopermissions', 'hvp'), 'NO_PERMISSION', 403);
|
|
return;
|
|
}
|
|
|
|
$token = required_param('_token', PARAM_RAW);
|
|
|
|
if (!H5PCore::validToken('contentHubRegistration', $token)) {
|
|
H5PCore::ajaxError('Invalid token', 'INVALID_TOKEN', 401);
|
|
return;
|
|
}
|
|
$logo = isset($_FILES['logo']) ? $_FILES['logo'] : null;
|
|
|
|
$formdata = [
|
|
'name' => required_param('name', PARAM_TEXT),
|
|
'email' => required_param('email', PARAM_EMAIL),
|
|
'description' => optional_param('description', '', PARAM_TEXT),
|
|
'contact_person' => optional_param('contact_person', '', PARAM_TEXT),
|
|
'phone' => optional_param('phone', '', PARAM_TEXT),
|
|
'address' => optional_param('address', '', PARAM_TEXT),
|
|
'city' => optional_param('city', '', PARAM_TEXT),
|
|
'zip' => optional_param('zip', '', PARAM_TEXT),
|
|
'country' => optional_param('country', '', PARAM_TEXT),
|
|
'remove_logo' => optional_param('remove_logo', '', PARAM_BOOL),
|
|
];
|
|
|
|
$core = framework::instance();
|
|
$result = $core->hubRegisterAccount($formdata, $logo);
|
|
|
|
if ($result['success'] === false) {
|
|
$core->h5pF->setErrorMessage($result['message']);
|
|
H5PCore::ajaxError($result['message'], $result['error_code'], $result['status_code']);
|
|
return;
|
|
}
|
|
|
|
$core->h5pF->setInfoMessage($result['message']);
|
|
H5PCore::ajaxSuccess($result['message']);
|
|
break;
|
|
|
|
/*
|
|
* Handle filtering of parameters through AJAX.
|
|
*/
|
|
case 'getcontent':
|
|
$token = required_param('token', PARAM_RAW);
|
|
$hubid = required_param('hubId', PARAM_INT);
|
|
|
|
$editor = \mod_hvp\framework::instance('editor');
|
|
$editor->ajax->action(H5PEditorEndpoints::GET_HUB_CONTENT, $token, $hubid, null);
|
|
break;
|
|
|
|
/*
|
|
* Handle publishing of content to the H5P OER Hub.
|
|
*/
|
|
case 'share':
|
|
if ($_SERVER['REQUEST_METHOD'] !== 'POST') {
|
|
http_response_code(405);
|
|
}
|
|
$token = required_param('_token', PARAM_RAW);
|
|
$id = required_param('id', PARAM_INT);
|
|
|
|
if (!\H5PCore::validToken('share_' . $id, $token)) {
|
|
\H5PCore::ajaxError(get_string('invalidtoken', 'hvp'));
|
|
break;
|
|
}
|
|
|
|
$data = array(
|
|
'title' => required_param('title', PARAM_RAW),
|
|
'language' => required_param('language', PARAM_RAW),
|
|
'license' => required_param('license', PARAM_RAW),
|
|
'license_version' => required_param('license_version', PARAM_RAW),
|
|
'level' => optional_param('level', null, PARAM_RAW),
|
|
'disciplines' => optional_param_array('disciplines', null, PARAM_RAW),
|
|
'keywords' => optional_param_array('keywords', null, PARAM_RAW),
|
|
'summary' => optional_param('summary', null, PARAM_RAW),
|
|
'description' => optional_param('description', null, PARAM_RAW),
|
|
'screenshot_alt_texts' => optional_param_array('screenshot_alt_texts', null, PARAM_RAW),
|
|
'remove_screenshots' => optional_param_array('remove_screenshots', null, PARAM_RAW),
|
|
'remove_icon' => optional_param('remove_icon', null, PARAM_RAW),
|
|
'age' => optional_param('age', null, PARAM_RAW),
|
|
);
|
|
|
|
// Load content.
|
|
$core = \mod_hvp\framework::instance();
|
|
$cm = get_coursemodule_from_id('hvp', $id);
|
|
$content = $core->loadContent($cm->instance);
|
|
|
|
// Update Hub status for content before proceeding.
|
|
$newstate = hvp_update_hub_status($content);
|
|
$synced = $newstate !== false ? $newstate : intval($content['synced']);
|
|
|
|
if ($synced === \H5PContentHubSyncStatus::WAITING) {
|
|
\H5PCore::ajaxError(get_string('contentissyncing', 'hvp'));
|
|
break;
|
|
}
|
|
|
|
// Create URL.
|
|
$data['download_url'] = hvp_create_hub_export_url($cm->id, $content);
|
|
|
|
// Get file size.
|
|
$slug = $content['slug'] ? $content['slug'] . '-' : '';
|
|
$filecontext = context_course::instance($cm->course);
|
|
$file = get_file_storage()->get_file($filecontext->id, 'mod_hvp', 'exports', 0, '/', "{$slug}{$content['id']}.h5p");
|
|
if (!$file) {
|
|
\H5PCore::ajaxError(get_string('noexport', 'hvp'));
|
|
break;
|
|
}
|
|
$size = $file->get_filesize();
|
|
$data['size'] = empty($size) ? -1 : $size;
|
|
|
|
// Add the icon and any screenshots.
|
|
$files = array(
|
|
'icon' => !empty($_FILES['icon']) ? $_FILES['icon'] : null,
|
|
'screenshots' => !empty($_FILES['screenshots']) ? $_FILES['screenshots'] : null,
|
|
);
|
|
|
|
try {
|
|
$isedit = !empty($content['contentHubId']);
|
|
$updatecontent = $synced === \H5PContentHubSyncStatus::NOT_SYNCED && $isedit;
|
|
if ($updatecontent) {
|
|
// Node has been edited since the last time it was published.
|
|
$data['resync'] = 1;
|
|
}
|
|
$result = $core->hubPublishContent($data, $files, $isedit ? $content['contentHubId'] : null);
|
|
|
|
$fields = array(
|
|
'shared' => 1, // Content is always shared after sharing or editing.
|
|
);
|
|
if (!$isedit) {
|
|
$fields['hub_id'] = $result->content->id;
|
|
// Sync will not happen on 'edit info', only for 'publish' or 'sync'.
|
|
$fields['synced'] = \H5PContentHubSyncStatus::WAITING;
|
|
} else if ($updatecontent) {
|
|
$fields['synced'] = \H5PContentHubSyncStatus::WAITING;
|
|
}
|
|
|
|
// Store the content hub id.
|
|
$core->h5pF->updateContentFields($cm->instance, $fields);
|
|
|
|
H5PCore::ajaxSuccess();
|
|
} catch (Exception $e) {
|
|
H5PCore::ajaxError(!empty($e->errors) ? $e->errors : $e->getMessage());
|
|
}
|
|
|
|
break;
|
|
|
|
/*
|
|
* Throw error if AJAX isnt handeled
|
|
*/
|
|
default:
|
|
throw new coding_exception('Unhandled AJAX');
|
|
break;
|
|
}
|