-
Notifications
You must be signed in to change notification settings - Fork 23
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #193 from learnweb/fix/draft-files
Fix and migrate draft-files to 'post' file area and fix messages not being filtered
- Loading branch information
Showing
7 changed files
with
398 additions
and
22 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,352 @@ | ||
<?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/>. | ||
|
||
/** | ||
* This file contains functions used by upgrade and install. | ||
* | ||
* Because this is used during install it should not include additional files. | ||
* | ||
* @package mod_moodleoverflow | ||
* @copyright 2024 Justus Dieckmann, University of Münster | ||
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later | ||
*/ | ||
|
||
// This is directly copied from filelib.php, but with a added $usercontext parameter. | ||
// phpcs:disable | ||
|
||
/** | ||
* Convert the draft file area URLs in some content to @@PLUGINFILE@@ tokens | ||
* ready to be saved in the database. Normally, this is done automatically by | ||
* {@link file_save_draft_area_files()}. | ||
* | ||
* @category files | ||
* @param string $text the content to process. | ||
* @param int $draftitemid the draft file area the content was using. | ||
* @param context_user $usercontext | ||
* @param bool $forcehttps whether the content contains https URLs. Default false. | ||
* @return string the processed content. | ||
*/ | ||
function mod_moodleoverflow_file_rewrite_urls_to_pluginfile($text, $draftitemid, $usercontext, $forcehttps = false) { | ||
global $CFG; | ||
|
||
$wwwroot = $CFG->wwwroot; | ||
if ($forcehttps) { | ||
$wwwroot = str_replace('http://', 'https://', $wwwroot); | ||
} | ||
|
||
// relink embedded files if text submitted - no absolute links allowed in database! | ||
$text = str_ireplace("$wwwroot/draftfile.php/$usercontext->id/user/draft/$draftitemid/", '@@PLUGINFILE@@/', $text); | ||
|
||
if (strpos($text, 'draftfile.php?file=') !== false) { | ||
$matches = array(); | ||
preg_match_all("!$wwwroot/draftfile.php\?file=%2F{$usercontext->id}%2Fuser%2Fdraft%2F{$draftitemid}%2F[^'\",&<>|`\s:\\\\]+!iu", $text, $matches); | ||
if ($matches) { | ||
foreach ($matches[0] as $match) { | ||
$replace = str_ireplace('%2F', '/', $match); | ||
$text = str_replace($match, $replace, $text); | ||
} | ||
} | ||
$text = str_ireplace("$wwwroot/draftfile.php?file=/$usercontext->id/user/draft/$draftitemid/", '@@PLUGINFILE@@/', $text); | ||
} | ||
|
||
return $text; | ||
} | ||
|
||
/** | ||
* Saves files from a draft file area to a real one (merging the list of files). | ||
* Can rewrite URLs in some content at the same time if desired. | ||
* | ||
* @category files | ||
* @param int $draftitemid the id of the draft area to use. Normally obtained | ||
* from file_get_submitted_draft_itemid('elementname') or similar. | ||
* When set to -1 (probably, by a WebService) it won't process file merging, keeping the original state of the file area. | ||
* @param int $contextid This parameter and the next two identify the file area to save to. | ||
* @param string $component | ||
* @param string $filearea indentifies the file area. | ||
* @param int $itemid helps identifies the file area. | ||
* @param context_user $usercontext | ||
* @param array $options area options (subdirs=>false, maxfiles=-1, maxbytes=0) | ||
* @param string $text some html content that needs to have embedded links rewritten | ||
* to the @@PLUGINFILE@@ form for saving in the database. | ||
* @param bool $forcehttps force https urls. | ||
* @return string|null if $text was passed in, the rewritten $text is returned. Otherwise NULL. | ||
*/ | ||
function mod_moodleoverflow_file_save_draft_area_files($draftitemid, $contextid, $component, $filearea, $itemid, $usercontext, array $options=null, $text=null, $forcehttps=false) { | ||
|
||
// Do not merge files, leave it as it was. | ||
if ($draftitemid === IGNORE_FILE_MERGE) { | ||
// Safely return $text, no need to rewrite pluginfile because this is mostly comming from an external client like the app. | ||
return $text; | ||
} | ||
|
||
if ($itemid === false) { | ||
// Catch a potentially dangerous coding error. | ||
throw new coding_exception('file_save_draft_area_files was called with $itemid false. ' . | ||
"This suggests a bug, because it would wipe all ($contextid, $component, $filearea) files."); | ||
} | ||
|
||
$fs = get_file_storage(); | ||
|
||
$options = (array)$options; | ||
if (!isset($options['subdirs'])) { | ||
$options['subdirs'] = false; | ||
} | ||
if (!isset($options['maxfiles'])) { | ||
$options['maxfiles'] = -1; // unlimited | ||
} | ||
if (!isset($options['maxbytes']) || $options['maxbytes'] == USER_CAN_IGNORE_FILE_SIZE_LIMITS) { | ||
$options['maxbytes'] = 0; // unlimited | ||
} | ||
if (!isset($options['areamaxbytes'])) { | ||
$options['areamaxbytes'] = FILE_AREA_MAX_BYTES_UNLIMITED; // Unlimited. | ||
} | ||
$allowreferences = true; | ||
if (isset($options['return_types']) && !($options['return_types'] & (FILE_REFERENCE | FILE_CONTROLLED_LINK))) { | ||
// we assume that if $options['return_types'] is NOT specified, we DO allow references. | ||
// this is not exactly right. BUT there are many places in code where filemanager options | ||
// are not passed to file_save_draft_area_files() | ||
$allowreferences = false; | ||
} | ||
|
||
// Check if the user has copy-pasted from other draft areas. Those files will be located in different draft | ||
// areas and need to be copied into the current draft area. | ||
$text = file_merge_draft_areas($draftitemid, $usercontext->id, $text, $forcehttps); | ||
|
||
// Check if the draft area has exceeded the authorised limit. This should never happen as validation | ||
// should have taken place before, unless the user is doing something nauthly. If so, let's just not save | ||
// anything at all in the next area. | ||
if (file_is_draft_area_limit_reached($draftitemid, $options['areamaxbytes'])) { | ||
return null; | ||
} | ||
|
||
$draftfiles = $fs->get_area_files($usercontext->id, 'user', 'draft', $draftitemid, 'id'); | ||
$oldfiles = $fs->get_area_files($contextid, $component, $filearea, $itemid, 'id'); | ||
|
||
// One file in filearea means it is empty (it has only top-level directory '.'). | ||
if (count($draftfiles) > 1 || count($oldfiles) > 1) { | ||
// we have to merge old and new files - we want to keep file ids for files that were not changed | ||
// we change time modified for all new and changed files, we keep time created as is | ||
|
||
$newhashes = array(); | ||
$filecount = 0; | ||
$context = context::instance_by_id($contextid, MUST_EXIST); | ||
foreach ($draftfiles as $file) { | ||
if (!$options['subdirs'] && $file->get_filepath() !== '/') { | ||
continue; | ||
} | ||
if (!$allowreferences && $file->is_external_file()) { | ||
continue; | ||
} | ||
if (!$file->is_directory()) { | ||
// Check to see if this file was uploaded by someone who can ignore the file size limits. | ||
$fileusermaxbytes = get_user_max_upload_file_size($context, $options['maxbytes'], 0, 0, $file->get_userid()); | ||
if ($fileusermaxbytes != USER_CAN_IGNORE_FILE_SIZE_LIMITS | ||
&& ($options['maxbytes'] and $options['maxbytes'] < $file->get_filesize())) { | ||
// Oversized file. | ||
continue; | ||
} | ||
if ($options['maxfiles'] != -1 and $options['maxfiles'] <= $filecount) { | ||
// more files - should not get here at all | ||
continue; | ||
} | ||
$filecount++; | ||
} | ||
$newhash = $fs->get_pathname_hash($contextid, $component, $filearea, $itemid, $file->get_filepath(), $file->get_filename()); | ||
$newhashes[$newhash] = $file; | ||
} | ||
|
||
// Loop through oldfiles and decide which we need to delete and which to update. | ||
// After this cycle the array $newhashes will only contain the files that need to be added. | ||
foreach ($oldfiles as $oldfile) { | ||
$oldhash = $oldfile->get_pathnamehash(); | ||
if (!isset($newhashes[$oldhash])) { | ||
// delete files not needed any more - deleted by user | ||
$oldfile->delete(); | ||
continue; | ||
} | ||
|
||
$newfile = $newhashes[$oldhash]; | ||
// Now we know that we have $oldfile and $newfile for the same path. | ||
// Let's check if we can update this file or we need to delete and create. | ||
if ($newfile->is_directory()) { | ||
// Directories are always ok to just update. | ||
} else if (($source = @unserialize($newfile->get_source() ?? '')) && isset($source->original)) { | ||
// File has the 'original' - we need to update the file (it may even have not been changed at all). | ||
$original = file_storage::unpack_reference($source->original); | ||
if ($original['filename'] !== $oldfile->get_filename() || $original['filepath'] !== $oldfile->get_filepath()) { | ||
// Very odd, original points to another file. Delete and create file. | ||
$oldfile->delete(); | ||
continue; | ||
} | ||
} else { | ||
// The same file name but absence of 'original' means that file was deteled and uploaded again. | ||
// By deleting and creating new file we properly manage all existing references. | ||
$oldfile->delete(); | ||
continue; | ||
} | ||
|
||
// status changed, we delete old file, and create a new one | ||
if ($oldfile->get_status() != $newfile->get_status()) { | ||
// file was changed, use updated with new timemodified data | ||
$oldfile->delete(); | ||
// This file will be added later | ||
continue; | ||
} | ||
|
||
// Updated author | ||
if ($oldfile->get_author() != $newfile->get_author()) { | ||
$oldfile->set_author($newfile->get_author()); | ||
} | ||
// Updated license | ||
if ($oldfile->get_license() != $newfile->get_license()) { | ||
$oldfile->set_license($newfile->get_license()); | ||
} | ||
|
||
// Updated file source | ||
// Field files.source for draftarea files contains serialised object with source and original information. | ||
// We only store the source part of it for non-draft file area. | ||
$newsource = $newfile->get_source(); | ||
if ($source = @unserialize($newfile->get_source() ?? '')) { | ||
$newsource = $source->source; | ||
} | ||
if ($oldfile->get_source() !== $newsource) { | ||
$oldfile->set_source($newsource); | ||
} | ||
|
||
// Updated sort order | ||
if ($oldfile->get_sortorder() != $newfile->get_sortorder()) { | ||
$oldfile->set_sortorder($newfile->get_sortorder()); | ||
} | ||
|
||
// Update file timemodified | ||
if ($oldfile->get_timemodified() != $newfile->get_timemodified()) { | ||
$oldfile->set_timemodified($newfile->get_timemodified()); | ||
} | ||
|
||
// Replaced file content | ||
if (!$oldfile->is_directory() && | ||
($oldfile->get_contenthash() != $newfile->get_contenthash() || | ||
$oldfile->get_filesize() != $newfile->get_filesize() || | ||
$oldfile->get_referencefileid() != $newfile->get_referencefileid() || | ||
$oldfile->get_userid() != $newfile->get_userid())) { | ||
$oldfile->replace_file_with($newfile); | ||
} | ||
|
||
// unchanged file or directory - we keep it as is | ||
unset($newhashes[$oldhash]); | ||
} | ||
|
||
// Add fresh file or the file which has changed status | ||
// the size and subdirectory tests are extra safety only, the UI should prevent it | ||
foreach ($newhashes as $file) { | ||
$file_record = array('contextid'=>$contextid, 'component'=>$component, 'filearea'=>$filearea, 'itemid'=>$itemid, 'timemodified'=>time()); | ||
if ($source = @unserialize($file->get_source() ?? '')) { | ||
// Field files.source for draftarea files contains serialised object with source and original information. | ||
// We only store the source part of it for non-draft file area. | ||
$file_record['source'] = $source->source; | ||
} | ||
|
||
if ($file->is_external_file()) { | ||
$repoid = $file->get_repository_id(); | ||
if (!empty($repoid)) { | ||
$context = context::instance_by_id($contextid, MUST_EXIST); | ||
$repo = repository::get_repository_by_id($repoid, $context); | ||
if (!empty($options)) { | ||
$repo->options = $options; | ||
} | ||
$file_record['repositoryid'] = $repoid; | ||
// This hook gives the repo a place to do some house cleaning, and update the $reference before it's saved | ||
// to the file store. E.g. transfer ownership of the file to a system account etc. | ||
$reference = $repo->reference_file_selected($file->get_reference(), $context, $component, $filearea, $itemid); | ||
|
||
$file_record['reference'] = $reference; | ||
} | ||
} | ||
|
||
$fs->create_file_from_storedfile($file_record, $file); | ||
} | ||
} | ||
|
||
// note: do not purge the draft area - we clean up areas later in cron, | ||
// the reason is that user might press submit twice and they would loose the files, | ||
// also sometimes we might want to use hacks that save files into two different areas | ||
|
||
if (is_null($text)) { | ||
return null; | ||
} else { | ||
return mod_moodleoverflow_file_rewrite_urls_to_pluginfile($text, $draftitemid, $usercontext, $forcehttps); | ||
} | ||
} | ||
// phpcs:enable | ||
|
||
/** | ||
* This is a function that checks all moodleoverflow posts for links to user's draft files | ||
* and moves them to the appropriate file area. | ||
* @return int amount of changed files. | ||
*/ | ||
function mod_moodleoverflow_move_draftfiles_to_permanent_filearea() { | ||
global $CFG, $DB; | ||
|
||
$baseurl = $CFG->wwwroot . '/draftfile.php/'; | ||
|
||
$posts = $DB->get_recordset_select( | ||
'moodleoverflow_posts', | ||
$DB->sql_like('message', ':searchurl'), | ||
['searchurl' => '%' . $DB->sql_like_escape($baseurl) . '%'], | ||
'discussion' | ||
); | ||
|
||
$discussion = null; | ||
$fs = get_file_storage(); | ||
|
||
$postsupdated = 0; | ||
|
||
foreach ($posts as $post) { | ||
|
||
if (!$discussion || $discussion->id != $post->discussion) { | ||
$discussion = $DB->get_record('moodleoverflow_discussions', ['id' => $post->discussion]); | ||
$moodleoverflow = $DB->get_record('moodleoverflow', ['id' => $discussion->moodleoverflow]); | ||
$cm = get_coursemodule_from_instance('moodleoverflow', $moodleoverflow->id); | ||
$context = context_module::instance($cm->id); | ||
} | ||
|
||
// Url is for example http://localhost/moodle404/draftfile.php/5/user/draft/42741688/Screenshot.png . | ||
$usercontext = context_user::instance($post->userid); | ||
$userurl = $baseurl . $usercontext->id . '/user/draft/'; | ||
$userurllen = strlen($userurl); | ||
|
||
$index = strpos($post->message, $userurl); | ||
if ($index === false) { | ||
continue; | ||
} | ||
|
||
$offset = strpos($post->message, '/', $index + $userurllen); | ||
$draftid = substr($post->message, $index + $userurllen, $offset - ($index + $userurllen)); | ||
if (!is_number($draftid) || $fs->is_area_empty($usercontext->id, 'user', 'draft', $draftid)) { | ||
continue; | ||
} | ||
|
||
$post->message = mod_moodleoverflow_file_save_draft_area_files($draftid, $context->id, 'mod_moodleoverflow', 'post', | ||
$post->id, $usercontext, mod_forum_post_form::editor_options($context, null), $post->message); | ||
|
||
$DB->set_field('moodleoverflow_posts', 'message', $post->message, ['id' => $post->id]); | ||
|
||
$postsupdated++; | ||
} | ||
|
||
return $postsupdated; | ||
|
||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.