Skip to content
Snippets Groups Projects
Commit b24d9195 authored by Andreas Gohr's avatar Andreas Gohr
Browse files

refactor page saving and introduce COMMON_WIKIPAGE_SAVE

This makes the saveWikiText() function a little easier to read and moves
external edit handling to its own function. Behavior stays the same
(tests are unchanged).

In addition a new event COMMON_WIKIPAGE_SAVE is introduced that makes
intercepting and acting on page saves much easier than possible before.

Developers can:

* prevent saves by either preventing the default action or overwriting
  the contentChanged field in a BEFORE hook
* enforce saves even when no content changed by overwriting the
  contentChanged field in a BEFORE hook
* Adjust the saved content by modifying the newContent field in a BEFORE
  hook
* Adjust the stored change log information (summary, type, extras) in an
  AFTER hook
* Easily know if a page was deleted, created or edited by inspecting the
  changeType field
* what ever they want before or after a wiki page is saved
parent 023953f0
No related branches found
No related tags found
No related merge requests found
...@@ -1183,6 +1183,38 @@ function con($pre, $text, $suf, $pretty = false) { ...@@ -1183,6 +1183,38 @@ function con($pre, $text, $suf, $pretty = false) {
return $pre.$text.$suf; return $pre.$text.$suf;
} }
/**
* Checks if the current page version is newer than the last entry in the page's
* changelog. If so, we assume it has been an external edit and we create an
* attic copy and add a proper changelog line.
*
* This check is only executed when the page is about to be saved again from the
* wiki, triggered in @see saveWikiText()
*
* @param string $id the page ID
*/
function detectExternalEdit($id) {
global $lang;
$file = wikiFN($id);
$old = @filemtime($file); // from page
$pagelog = new PageChangeLog($id, 1024);
$oldRev = $pagelog->getRevisions(-1, 1); // from changelog
$oldRev = (int) (empty($oldRev) ? 0 : $oldRev[0]);
if(!file_exists(wikiFN($id, $old)) && file_exists($file) && $old >= $oldRev) {
// add old revision to the attic if missing
saveOldRevision($id);
// add a changelog entry if this edit came from outside dokuwiki
if($old > $oldRev) {
addLogEntry($old, $id, DOKU_CHANGE_TYPE_EDIT, $lang['external_edit'], '', array('ExternalEdit'=> true));
// remove soon to be stale instructions
$cache = new cache_instructions($id, $file);
$cache->removeCache();
}
}
}
/** /**
* Saves a wikitext by calling io_writeWikiPage. * Saves a wikitext by calling io_writeWikiPage.
* Also directs changelog and attic updates. * Also directs changelog and attic updates.
...@@ -1208,77 +1240,72 @@ function saveWikiText($id, $text, $summary, $minor = false) { ...@@ -1208,77 +1240,72 @@ function saveWikiText($id, $text, $summary, $minor = false) {
/* @var Input $INPUT */ /* @var Input $INPUT */
global $INPUT; global $INPUT;
// ignore if no changes were made // prepare data for event
if($text == rawWiki($id, '')) { $svdta = array();
return; $svdta['id'] = $id;
} $svdta['file'] = wikiFN($id);
$svdta['revertFrom'] = $REV;
$svdta['oldRevision'] = @filemtime($svdta['file']);
$svdta['newRevision'] = 0;
$svdta['newContent'] = $text;
$svdta['oldContent'] = rawWiki($id);
$svdta['summary'] = $summary;
$svdta['contentChanged'] = ($svdta['newContent'] != $svdta['oldContent']);
$svdta['changeInfo'] = '';
$svdta['changeType'] = DOKU_CHANGE_TYPE_EDIT;
$file = wikiFN($id); // select changelog line type
$old = @filemtime($file); // from page if($REV) {
$wasRemoved = (trim($text) == ''); // check for empty or whitespace only $svdta['changeType'] = DOKU_CHANGE_TYPE_REVERT;
$wasCreated = !file_exists($file); $svdta['changeInfo'] = $REV;
$wasReverted = ($REV == true); } else if(!file_exists($svdta['file'])) {
$pagelog = new PageChangeLog($id, 1024); $svdta['changeType'] = DOKU_CHANGE_TYPE_CREATE;
$newRev = false; } else if(trim($text) == '') {
$oldRev = $pagelog->getRevisions(-1, 1); // from changelog // empty or whitespace only content deletes
$oldRev = (int) (empty($oldRev) ? 0 : $oldRev[0]); $svdta['changeType'] = DOKU_CHANGE_TYPE_DELETE;
if(!file_exists(wikiFN($id, $old)) && file_exists($file) && $old >= $oldRev) { // autoset summary on deletion
// add old revision to the attic if missing if(blank($svdta['summary'])) $svdta['summary'] = $lang['deleted'];
saveOldRevision($id); } else if($minor && $conf['useacl'] && $INPUT->server->str('REMOTE_USER')) {
// add a changelog entry if this edit came from outside dokuwiki //minor edits only for logged in users
if($old > $oldRev) { $svdta['changeType'] = DOKU_CHANGE_TYPE_MINOR_EDIT;
addLogEntry($old, $id, DOKU_CHANGE_TYPE_EDIT, $lang['external_edit'], '', array('ExternalEdit'=> true));
// remove soon to be stale instructions
$cache = new cache_instructions($id, $file);
$cache->removeCache();
}
} }
if($wasRemoved) { $event = new Doku_Event('COMMON_WIKIPAGE_SAVE', $svdta);
if(!$event->advise_before()) return;
// if the content has not been changed, no save happens (plugins may override this)
if(!$svdta['contentChanged']) return;
detectExternalEdit($id);
if($svdta['changeType'] == DOKU_CHANGE_TYPE_DELETE) {
// Send "update" event with empty data, so plugins can react to page deletion // Send "update" event with empty data, so plugins can react to page deletion
$data = array(array($file, '', false), getNS($id), noNS($id), false); $data = array(array($svdta['file'], '', false), getNS($id), noNS($id), false);
trigger_event('IO_WIKIPAGE_WRITE', $data); trigger_event('IO_WIKIPAGE_WRITE', $data);
// pre-save deleted revision // pre-save deleted revision
@touch($file); @touch($svdta['file']);
clearstatcache(); clearstatcache();
$newRev = saveOldRevision($id); $data['newRevision'] = saveOldRevision($id);
// remove empty file // remove empty file
@unlink($file); @unlink($svdta['file']);
// don't remove old meta info as it should be saved, plugins can use IO_WIKIPAGE_WRITE for removing their metadata... // don't remove old meta info as it should be saved, plugins can use IO_WIKIPAGE_WRITE for removing their metadata...
// purge non-persistant meta data // purge non-persistant meta data
p_purge_metadata($id); p_purge_metadata($id);
$del = true;
// autoset summary on deletion
if(empty($summary)) $summary = $lang['deleted'];
// remove empty namespaces // remove empty namespaces
io_sweepNS($id, 'datadir'); io_sweepNS($id, 'datadir');
io_sweepNS($id, 'mediadir'); io_sweepNS($id, 'mediadir');
} else { } else {
// save file (namespace dir is created in io_writeWikiPage) // save file (namespace dir is created in io_writeWikiPage)
io_writeWikiPage($file, $text, $id); io_writeWikiPage($svdta['file'], $text, $id);
// pre-save the revision, to keep the attic in sync // pre-save the revision, to keep the attic in sync
$newRev = saveOldRevision($id); $svdta['newRevision'] = saveOldRevision($id);
$del = false;
} }
// select changelog line type $event->advise_after();
$extra = '';
$type = DOKU_CHANGE_TYPE_EDIT;
if($wasReverted) {
$type = DOKU_CHANGE_TYPE_REVERT;
$extra = $REV;
} else if($wasCreated) {
$type = DOKU_CHANGE_TYPE_CREATE;
} else if($wasRemoved) {
$type = DOKU_CHANGE_TYPE_DELETE;
} else if($minor && $conf['useacl'] && $INPUT->server->str('REMOTE_USER')) {
$type = DOKU_CHANGE_TYPE_MINOR_EDIT;
} //minor edits only for logged in users
addLogEntry($newRev, $id, $type, $summary, $extra); addLogEntry($svdta['newRevision'], $svdta['id'], $svdta['changeType'], $svdta['summary'], $svdta['changeInfo']);
// send notify mails // send notify mails
notify($id, 'admin', $old, $summary, $minor); notify($svdta['id'], 'admin', $svdta['oldRevision'], $svdta['summary'], $minor);
notify($id, 'subscribers', $old, $summary, $minor); notify($svdta['id'], 'subscribers', $svdta['oldRevision'], $svdta['summary'], $minor);
// update the purgefile (timestamp of the last time anything within the wiki was changed) // update the purgefile (timestamp of the last time anything within the wiki was changed)
io_saveFile($conf['cachedir'].'/purgefile', time()); io_saveFile($conf['cachedir'].'/purgefile', time());
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment