Beispiel #1
0
 public function savePageEvent($runData)
 {
     $pl = $runData->getParameterList();
     $pageId = $pl->getParameterValue("page_id");
     $mode = $pl->getParameterValue("mode");
     if ($pl->getParameterValue("form")) {
         $data = array();
         $newpages = array();
         foreach ($runData->getParameterList()->asArray() as $name => $val) {
             $m = array();
             if (preg_match("/^field_(.*)\$/", $name, $m)) {
                 $data[$m[1]] = $val;
             }
         }
         $source = Wikidot_Yaml::dump($data);
     } else {
         $source = trim($pl->getParameterValue("source"));
     }
     $comments = trim($pl->getParameterValue("comments"));
     $title = trim($pl->getParameterValue("title"));
     $userId = $runData->getUserId();
     if ($userId == null) {
         $userString = $runData->createIpString();
     }
     if ($title === '') {
         $title = null;
     }
     $unixName = $pl->getParameterValue("wiki_page");
     $unixName = WDStringUtils::toUnixName($unixName);
     // purify! (for sure)
     $lockId = $pl->getParameterValue("lock_id");
     $lockSecret = $pl->getParameterValue("lock_secret");
     $site = $runData->getTemp("site");
     // validate input first
     $db = Database::connection();
     $db->begin();
     // remove old locks.
     if (strlen8($title) > 128) {
         throw new ProcessException(_("Title of the page should not be longer than 128 characters."), "title_too_long");
     }
     // if page source not too long...
     if (strlen8($source) > 200000) {
         throw new ProcessException(_("Source of the page should not be longer than 200 000 characters which is large enough. Pages longer than that can indicate improper usage \tof the wiki site."), "source_too_long");
     }
     // if comment too long
     if (strlen8($comments) > 210) {
         throw new ProcessException(_("The changes comment is longer than 200 characters. Please keep this description short and informative. And no longer than this limit please..."), "comment_too_long");
     }
     $autoincrement = false;
     $nowDate = new ODate();
     if ($pageId === null || $pageId === '') {
         if (preg_match(';^([a-z0-9]+:)?' . self::$AUTOINCREMENT_PAGE . '$;', $unixName)) {
             $autoincrement = true;
         }
         if (!$autoincrement) {
             DB_PageEditLockPeer::instance()->deleteOutdatedByPageName($site->getSiteId(), $unixName);
         }
         // a page should be created!
         // extract category name
         if (strpos($unixName, ':') != false) {
             // ok, there is category!
             $exp = explode(':', $unixName);
             $categoryName = $exp[0];
         } else {
             // no category name, "_default" assumed
             $categoryName = "_default";
         }
         // check if category exists. if not - create it!
         $category = DB_CategoryPeer::instance()->selectByName($categoryName, $site->getSiteId(), false);
         if ($category == null) {
             // create the category - just clone the default category!!!
             $category = DB_CategoryPeer::instance()->selectByName("_default", $site->getSiteId(), false);
             $category->setName($categoryName);
             // fill with some important things - we assume the _default category exists!!! IT REALLY SHOULD!!!
             $category->setCategoryId(null);
             $category->setNew(true);
             // this will make it INSERT, not UPDATE on save()
             $category->setPerPageDiscussion(null);
             //default value
             // set default permissions theme and license
             $category->setPermissionsDefault(true);
             $category->setThemeDefault(true);
             $category->setLicenseDefault(true);
             $category->setNavDefault(true);
             $category->save();
         }
         // first look at permissions!
         WDPermissionManager::instance()->hasPagePermission('create', $runData->getUser(), $category);
         // check the locks!
         // check if the lock still exists.
         if (!$autoincrement) {
             $c = new Criteria();
             $c->add("lock_id", $lockId);
             $c->add("secret", $lockSecret);
             $lock = DB_PageEditLockPeer::instance()->selectOne($c);
             if ($lock == null) {
                 $page = DB_PagePeer::instance()->selectByName($site->getSiteId(), $unixName);
                 if ($page != null) {
                     // page exists!!! error!
                     $runData->ajaxResponseAdd("noLockError", "other_locks");
                     $runData->ajaxResponseAdd("pageExists", true);
                     $runData->ajaxResponseAdd("locked", true);
                     //well, it is somehow locked...
                     $runData->setModuleTemplate("edit/NewPageExistsWinModule");
                     $runData->contextAdd("nonrecoverable", true);
                     $runData->ajaxResponseAdd("nonrecoverable", true);
                     $db->commit();
                     return;
                 }
                 // check if we can TRANSPARENTLY recreate the lock IF there is no
                 // conflicting lock and the revision_id has not changed.
                 $lock = new DB_PageEditLock();
                 $lock->setPageUnixName($unixName);
                 $lock->setSiteId($site->getSiteId());
                 $lock->setUserId($runData->getUserId());
                 $lock->setUserString($runData->getSession()->getIpAddress());
                 $lock->setDateStarted(new ODate());
                 $lock->setDateLastAccessed(new ODate());
                 $lock->setMode("page");
                 $conflictLocks = $lock->getConflicts();
                 if ($conflictLocks == null) {
                     // safely recreate lock
                     $secret = md5(time() . rand(1000, 9999));
                     $lock->setSecret($secret);
                     $lock->setSessionId($runData->getSession()->getSessionId());
                     $lock->save();
                     $lockId = $lock->getLockId();
                     // send back new lock information
                     $runData->ajaxResponseAdd("lockRecreated", true);
                     $runData->ajaxResponseAdd("lockId", $lockId);
                     $runData->ajaxResponseAdd("lockSecret", $secret);
                     $runData->ajaxResponseAdd('timeLeft', 60 * 15);
                 } else {
                     $runData->ajaxResponseAdd("noLockError", "other_locks");
                     $runData->setModuleTemplate("edit/LockInterceptedWinModule");
                     $runData->contextAdd("locks", $conflictLocks);
                     $db->commit();
                     return;
                 }
             } else {
                 $lock->setDateLastAccessed(new ODate());
                 $lock->save();
                 $runData->ajaxResponseAdd('timeLeft', 60 * 15);
             }
         }
         /* Change unixName to integer. */
         if ($autoincrement) {
             /* Check max number taken. */
             $db = Database::connection();
             $q = "select max(substring(unix_name from '[0-9]+')::integer) + 1 as max from page where category_id={$category->getCategoryId()} AND unix_name ~ '^([a-z0-9]+:)?[0-9]+\$'";
             $r = $db->query($q);
             $row = $r->nextRow();
             $unixName = $row['max'];
             if ($category->getName() != '_default') {
                 $unixName = $category->getName() . ':' . $unixName;
             }
             $runData->ajaxResponseAdd('pageUnixName', $unixName);
         }
         $page = new DB_Page();
         $page->obtainPK();
         $pageRevision = new DB_PageRevision();
         $pageRevision->setSiteId($site->getSiteId());
         $pageRevision->setPageId($page->getPageId());
         $pageRevision->setFlagNew(true);
         $pageRevision->setComments($comments);
         $pageRevision->obtainPK();
         $pageRevision->setDateLastEdited($nowDate);
         $pageRevision->setPageId($page->getPageId());
         $page->setRevisionId($pageRevision->getRevisionId());
         $pageSource = new DB_PageSource();
         $pageSource->setText($source);
         $pageSource->save();
         $pageRevision->setSourceId($pageSource->getSourceId());
         $pageMetadata = new DB_PageMetadata();
         $pageMetadata->setTitle($title);
         $pageMetadata->setUnixName($unixName);
         if ($userId) {
             $pageMetadata->setOwnerUserId($userId);
         }
         $pageMetadata->save();
         $pageRevision->setMetadataId($pageMetadata->getMetadataId());
         $pageCompiled = new DB_PageCompiled();
         $pageCompiled->setPageId($page->getPageId());
         $newPage = true;
         // update the page object
         $page->setUnixName($unixName);
         $page->setDateCreated($nowDate);
         $page->setSiteId($site->getSiteId());
         $page->setSourceId($pageSource->getSourceId());
         $page->setMetadataId($pageMetadata->getMetadataId());
         $page->setTitle($title);
         $page->setDateLastEdited($nowDate);
         $pageCompiled = new DB_PageCompiled();
         $pageCompiled->setPageId($page->getPageId());
         $pageCompiled->outdate();
         $newPage = true;
         $page->setCategoryId($category->getCategoryId());
         // now set user_id, user_string
         if ($userId) {
             $pageRevision->setUserId($userId);
             $page->setLastEditUserId($userId);
         } else {
             $pageRevision->setUserId(0);
             $page->setLastEditUserId(0);
             $pageRevision->setUserString($userString);
             $page->setLastEditUserString($userString);
         }
         $page->setOwnerUserId($userId);
         $pageRevision->save();
         $page->setRevisionId($pageRevision->getRevisionId());
         $page->save();
         $pageCompiled->save();
         $sourceChanged = true;
         $outdater = new Outdater();
         $outdater->pageEvent("new_page", $page);
         // index page
         if (!$autoincrement) {
             $c = new Criteria();
             $c->add("lock_id", $lockId);
             DB_PageEditLockPeer::instance()->delete($c);
         }
         EventLogger::instance()->logNewPage($page);
     } else {
         // THE PAGE ALREADY EXISTS
         DB_PageEditLockPeer::instance()->deleteOutdated($pageId);
         $c = new Criteria();
         $c->add("page_id", $pageId);
         $c->setForUpdate(true);
         $page = DB_PagePeer::instance()->selectOne($c);
         if ($page == null) {
             throw new ProcessException(_("Page does not exist."));
         }
         // check permissions
         $category = $page->getCategory();
         WDPermissionManager::instance()->hasPagePermission('edit', $runData->getUser(), $category, $page);
         // check if the lock still exists.
         $c = new Criteria();
         $c->add("lock_id", $lockId);
         $c->add("secret", $lockSecret);
         $lock = DB_PageEditLockPeer::instance()->selectOne($c);
         if ($lock == null) {
             OzoneLogger::instance()->debug("no lock");
             // no lock!!! not good.
             if ($page->getRevisionId() != $pl->getParameterValue("revision_id")) {
                 // this is nonrecoverable.
                 // author should stop editing now!!!
                 OzoneLogger::instance()->debug("page changed");
                 $runData->ajaxResponseAdd("noLockError", "page_changed");
                 $runData->setModuleTemplate("edit/LockInterceptedWinModule");
                 $runData->contextAdd("nonrecoverable", true);
                 $runData->ajaxResponseAdd("nonrecoverable", true);
                 $db->commit();
                 return;
             }
             // check if we can TRANSPARENTLY recreate the lock IF there is no
             // conflicting lock and the revision_id has not changed.
             $lock = new DB_PageEditLock();
             $lock->setPageId($page->getPageId());
             $lock->setPageUnixName($page->getUnixName());
             $lock->setSiteId($site->getSiteId());
             $lock->setUserId($runData->getUserId());
             $lock->setUserString($runData->getSession()->getIpAddress());
             $lock->setDateStarted(new ODate());
             $lock->setDateLastAccessed(new ODate());
             $lock->setMode($mode);
             if ($mode == "section") {
                 $rangeStart = $pl->getParameterValue("range_start");
                 $rangeEnd = $pl->getParameterValue("range_end");
                 $lock->setRangeStart($rangeStart);
                 $lock->setRangeEnd($rangeEnd);
             }
             $conflictLocks = $lock->getConflicts();
             if ($conflictLocks == null) {
                 // safely recreate lock
                 $secret = md5(time() . rand(1000, 9999));
                 $lock->setSecret($secret);
                 $lock->setSessionId($runData->getSession()->getSessionId());
                 $lock->save();
                 $lockId = $lock->getLockId();
                 // send back new lock information
                 $runData->ajaxResponseAdd("lockRecreated", true);
                 $runData->ajaxResponseAdd("lockId", $lockId);
                 $runData->ajaxResponseAdd("lockSecret", $secret);
                 $runData->ajaxResponseAdd('timeLeft', 60 * 15);
             } else {
                 $runData->ajaxResponseAdd("noLockError", "other_locks");
                 $runData->setModuleTemplate("edit/LockInterceptedWinModule");
                 $runData->contextAdd("locks", $conflictLocks);
                 $db->commit();
                 return;
             }
         } else {
             $lock->setDateLastAccessed(new ODate());
             $lock->save();
             $runData->ajaxResponseAdd('timeLeft', 60 * 15);
             // here is a good place to check conditions for
             // "save & continue" which when first called
             // creates new revision, but the subsequent calls
             // do not.
         }
         // check if source or metadata has changed. if neither is changed - do nothing
         // get current revision
         $currentRevision = $page->getCurrentRevision();
         // compare source text
         $oldSourceText = $page->getSource();
         $sourceChanged = false;
         if ($mode == "append") {
             $source = $oldSourceText . "\n\n" . $source;
         }
         if ($mode == "section") {
             $rangeStart = $lock->getRangeStart();
             //$pl->getParameterValue("range_start");
             $rangeEnd = $lock->getRangeEnd();
             //$pl->getParameterValue("range_end");
             $s2 = explode("\n", $oldSourceText);
             // fix source last empty line
             if (!ereg("\n\$", $source)) {
                 $source .= "\n";
             }
             array_splice($s2, $rangeStart, $rangeEnd - $rangeStart + 1, explode("\n", $source));
             $source = implode("\n", $s2);
         }
         if ($oldSourceText !== $source) {
             $sourceChanged = true;
         }
         // create new revision
         $pageRevision = new DB_PageRevision();
         $pageRevision->setSiteId($site->getSiteId());
         // compare metadata
         $metadataChanged = false;
         $oldMetadata = $page->getMetadata();
         // title
         if ($mode == 'page') {
             // check only if the whole page is edited
             if ($title !== $oldMetadata->getTitle()) {
                 $pageRevision->setFlagTitle(true);
                 $metadataChanged = true;
             }
         }
         // and act accordingly to the situation
         if ($sourceChanged == false && $metadataChanged == false) {
             $c = new Criteria();
             $c->add("lock_id", $lockId);
             DB_PageEditLockPeer::instance()->delete($c);
             $db->commit();
             return;
         }
         $pageRevision->setPageId($page->getPageId());
         $pageRevision->setDateLastEdited($nowDate);
         $pageRevision->setRevisionNumber($currentRevision->getRevisionNumber() + 1);
         if ($sourceChanged) {
             $fullSource = false;
             // first check if store new source as a diff or as a full-source.
             if (true || $currentRevision->getSinceFullSource() > 9) {
                 $fullSource = true;
             } else {
                 // also compare size of diff against size of new source.
                 // must be less than %50 to qualify
                 $differ = new ODiff();
                 $diff = $differ->diffString($oldSourceText, $source);
                 if (strlen($diff) > 0.5 * strlen($source)) {
                     $fullSource = true;
                 }
             }
             $pageSource = new DB_PageSource();
             if ($fullSource) {
                 $pageSource->setText($source);
             } else {
                 $pageSource->setText($diff);
                 $pageRevision->setDiffSource(true);
                 $pageRevision->setSinceFullSource($currentRevision->getSinceFullSource() + 1);
             }
             $pageSource->save();
             $pageRevision->setSourceId($pageSource->getSourceId());
             $pageRevision->setFlagText(true);
         } else {
             // copy source id
             $pageRevision->setSourceId($currentRevision->getSourceId());
             $pageRevision->setSinceFullSource($currentRevision->getSinceFullSource());
             $pageRevision->setDiffSource($currentRevision->getDiffSource());
         }
         if ($metadataChanged) {
             $pageMetadata = clone $oldMetadata;
             $pageMetadata->setNew(true);
             $pageMetadata->setMetadataId(null);
             $pageMetadata->setTitle($title);
             $pageMetadata->save();
             $pageRevision->setMetadataId($pageMetadata->getMetadataId());
         } else {
             // copy metadata id
             $pageRevision->setMetadataId($currentRevision->getMetadataId());
         }
         // now set user_id, user_string
         if ($userId) {
             $pageRevision->setUserId($userId);
             $page->setLastEditUserId($userId);
         } else {
             $pageRevision->setUserId(0);
             $page->setLastEditUserId(0);
             $pageRevision->setUserString($userString);
             $page->setLastEditUserString($userString);
         }
         $pageRevision->setComments($comments);
         $pageRevision->save();
         $page->setRevisionId($pageRevision->getRevisionId());
         // update Page object
         $page->setSourceId($pageRevision->getSourceId());
         if ($mode == 'page') {
             $page->setTitle($title);
         }
         $page->setDateLastEdited($nowDate);
         $page->setMetadataId($pageRevision->getMetadataId());
         $page->setRevisionNumber($pageRevision->getRevisionNumber());
         $page->save();
         // also if "section edit" - find other locks that refer to
         // blocks with higher line numbers and change start/end accordingly
         if ($mode == "section") {
             $c = new Criteria();
             $c->add("page_id", $pageId);
             $c->add("range_start", $lock->getRangeEnd(), ">=");
             $c->add("mode", "section");
             $laterLocks = DB_PageEditLockPeer::instance()->select($c);
             if (count($laterLocks) > 0) {
                 // take the length of the current lock
                 $sectionLength = $lock->getRangeEnd() - $lock->getRangeStart() + 1;
                 $newSourceLength = count(explode("\n", trim($pl->getParameterValue("source")))) + 1;
                 // +1 for the new line at the end
                 $lengthDifference = $newSourceLength - $sectionLength;
                 foreach ($laterLocks as $llock) {
                     $llock->setRangeStart($llock->getRangeStart() + $lengthDifference);
                     $llock->setRangeEnd($llock->getRangeEnd() + $lengthDifference);
                     $llock->save();
                 }
             }
         }
         // OUTDATING PARTY!!!
         $outdater = new Outdater();
         if ($sourceChanged) {
             $outdater->pageEvent("source_changed", $page);
         }
         if ($metadataChanged) {
             $outdater->pageEvent("title_changed", $page);
         }
         // index page
         EventLogger::instance()->logSavePage($page);
     }
     // remove lock too?
     if (!$pl->getParameterValue("and_continue") && !$autoincrement) {
         $c = new Criteria();
         $c->add("lock_id", $lockId);
         DB_PageEditLockPeer::instance()->delete($c);
         $runData->ajaxResponseAdd("revisionId", $pageRevision->getRevisionId());
     }
     $db->commit();
 }