diff --git a/conf/dokuwiki.php b/conf/dokuwiki.php
index 67d515b2960dc59dd96ad55d6c62a73d1125853d..54243102986a8d68003ac6f27834c5a89ceb0124 100644
--- a/conf/dokuwiki.php
+++ b/conf/dokuwiki.php
@@ -99,6 +99,7 @@ $conf['rss_linkto'] = 'diff';            //what page RSS entries link to:
                                          //  'rev'     - page showing all revisions
                                          //  'current' - most recent revision of page
 $conf['rss_update'] = 5*60;              //Update the RSS feed every n minutes (defaults to 5 minutes)
+$conf['recent_days'] = 7;                //How many days of recent changes to keep. (days)
 
 //Set target to use when creating links - leave empty for same window
 $conf['target']['wiki']      = '';
diff --git a/inc/common.php b/inc/common.php
index 3064c4fda28798e35b23d02b44a63c7ee7231d8a..a0e1e882b40a3e63eed23e60fd6f714cc06ec007 100644
--- a/inc/common.php
+++ b/inc/common.php
@@ -94,16 +94,20 @@ function pageinfo(){
   $info['editable']  = ($info['writable'] && empty($info['lock']));
   $info['lastmod']   = @filemtime($info['filepath']);
 
+  //load page meta data
+  $info['meta'] = p_get_metadata($ID);
+
   //who's the editor
   if($REV){
-    $revinfo = getRevisionInfo($ID,$REV,false);
+    $revinfo = getRevisionInfo($ID, $REV, 1024);
   }else{
-    $revinfo = getRevisionInfo($ID,$info['lastmod'],false);
+    $revinfo = $info['meta']['last_change'];
   }
   $info['ip']     = $revinfo['ip'];
   $info['user']   = $revinfo['user'];
   $info['sum']    = $revinfo['sum'];
-  $info['minor']  = $revinfo['minor'];
+  // See also $INFO['meta']['last_change'] which is the most recent log line for page $ID.
+  // Use $INFO['meta']['last_change']['type']==='e' in place of $info['minor'].
 
   if($revinfo['user']){
     $info['editor'] = $revinfo['user'];
@@ -710,46 +714,53 @@ function dbglog($msg){
 }
 
 /**
- * Add's an entry to the changelog
+ * Add's an entry to the changelog and saves the metadata for the page
  *
  * @author Andreas Gohr <andi@splitbrain.org>
+ * @author Esther Brunner <wikidesign@gmail.com>
+ * @author Ben Coburn <btcoburn@silicodon.net>
  */
-function addLogEntry($date,$id,$summary='',$minor=false){
-  global $conf;
+function addLogEntry($date, $id, $type='E', $summary='', $extra=''){
+  global $conf, $INFO;
 
-  if(!@is_writable($conf['changelog'])){
-    msg($conf['changelog'].' is not writable!',-1);
-    return;
-  }
+  $id = cleanid($id);
+  $file = wikiFN($id);
+  $created = @filectime($file);
+  $minor = ($type==='e');
+  $wasRemoved = ($type==='D');
 
   if(!$date) $date = time(); //use current time if none supplied
   $remote = $_SERVER['REMOTE_ADDR'];
   $user   = $_SERVER['REMOTE_USER'];
 
-  if($conf['useacl'] && $user && $minor){
-    $summary = '*'.$summary;
-  }else{
-    $summary = ' '.$summary;
+  $logline = array(
+    'date'  => $date,
+    'ip'    => $remote,
+    'type'  => $type,
+    'id'    => $id,
+    'user'  => $user,
+    'sum'   => $summary,
+    'extra' => $extra
+  );
+
+  // update metadata
+  if (!$wasRemoved) {
+    $meta = array();
+    if (!$INFO['exists']){ // newly created
+      $meta['date']['created'] = $created;
+      if ($user) $meta['creator'] = $INFO['userinfo']['name'];
+    } elseif (!$minor) {   // non-minor modification
+      $meta['date']['modified'] = $date;
+      if ($user) $meta['contributor'][$user] = $INFO['userinfo']['name'];
+    }
+    $meta['last_change'] = $logline;
+    p_set_metadata($id, $meta, true);
   }
 
-  $logline = join("\t",array($date,$remote,$id,$user,$summary))."\n";
-  io_saveFile($conf['changelog'],$logline,true);
-}
-
-/**
- * Checks an summary entry if it was a minor edit
- *
- * The summary is cleaned of the marker char
- *
- * @author Andreas Gohr <andi@splitbrain.org>
- */
-function isMinor(&$summary){
-  if(substr($summary,0,1) == '*'){
-    $summary = substr($summary,1);
-    return true;
-  }
-  $summary = trim($summary);
-  return false;
+  // add changelog lines
+  $logline = implode("\t", $logline)."\n";
+  io_saveFile(metaFN($id,'.changes'),$logline,true); //page changelog
+  io_saveFile($conf['changelog'],$logline,true); //global changelog cache
 }
 
 /**
@@ -759,58 +770,39 @@ function isMinor(&$summary){
  *
  * @see getRecents()
  * @author Andreas Gohr <andi@splitbrain.org>
+ * @author Ben Coburn <btcoburn@silicodon.net>
  */
 function _handleRecent($line,$ns,$flags){
   static $seen  = array();         //caches seen pages and skip them
   if(empty($line)) return false;   //skip empty lines
 
   // split the line into parts
-  list($dt,$ip,$id,$usr,$sum) = explode("\t",$line);
+  $recent = parseChangelogLine($line);
+  if ($recent===false) { return false; }
 
   // skip seen ones
-  if($seen[$id]) return false;
-  $recent = array();
+  if(isset($seen[$recent['id']])) return false;
 
-  // check minors
-  if(isMinor($sum)){
-    // skip minors
-    if($flags & RECENTS_SKIP_MINORS) return false;
-    $recent['minor'] = true;
-  }else{
-    $recent['minor'] = false;
-  }
+  // skip minors
+  if($recent['type']==='e' && ($flags & RECENTS_SKIP_MINORS)) return false;
 
   // remember in seen to skip additional sights
-  $seen[$id] = 1;
+  $seen[$recent['id']] = 1;
 
   // check if it's a hidden page
-  if(isHiddenPage($id)) return false;
+  if(isHiddenPage($recent['id'])) return false;
 
   // filter namespace
-  if (($ns) && (strpos($id,$ns.':') !== 0)) return false;
+  if (($ns) && (strpos($recent['id'],$ns.':') !== 0)) return false;
 
   // exclude subnamespaces
-  if (($flags & RECENTS_SKIP_SUBSPACES) && (getNS($id) != $ns)) return false;
+  if (($flags & RECENTS_SKIP_SUBSPACES) && (getNS($recent['id']) != $ns)) return false;
 
   // check ACL
-  if (auth_quickaclcheck($id) < AUTH_READ) return false;
+  if (auth_quickaclcheck($recent['id']) < AUTH_READ) return false;
 
   // check existance
-  if(!@file_exists(wikiFN($id))){
-    if($flags & RECENTS_SKIP_DELETED){
-      return false;
-    }else{
-      $recent['del'] = true;
-    }
-  }else{
-    $recent['del'] = false;
-  }
-
-  $recent['id']   = $id;
-  $recent['date'] = $dt;
-  $recent['ip']   = $ip;
-  $recent['user'] = $usr;
-  $recent['sum']  = $sum;
+  if((!@file_exists(wikiFN($recent['id']))) && ($flags & RECENTS_SKIP_DELETED)) return false;
 
   return $recent;
 }
@@ -832,7 +824,7 @@ function _handleRecent($line,$ns,$flags){
  * @param string $ns      restrict to given namespace
  * @param bool   $flags   see above
  *
- * @author Andreas Gohr <andi@splitbrain.org>
+ * @author Ben Coburn <btcoburn@silicodon.net>
  */
 function getRecents($first,$num,$ns='',$flags=0){
   global $conf;
@@ -842,190 +834,245 @@ function getRecents($first,$num,$ns='',$flags=0){
   if(!$num)
     return $recent;
 
-  if(!@is_readable($conf['changelog'])){
-    msg($conf['changelog'].' is not readable',-1);
-    return $recent;
-  }
-
-  $fh  = fopen($conf['changelog'],'r');
-  $buf = '';
-  $csz = 4096;                              //chunksize
-  fseek($fh,0,SEEK_END);                    // jump to the end
-  $pos = ftell($fh);                        // position pointer
-
-  // now read backwards into buffer
-  while($pos > 0){
-    $pos -= $csz;                           // seek to previous chunk...
-    if($pos < 0) {                          // ...or rest of file
-      $csz += $pos;
-      $pos = 0;
+  // read all recent changes. (kept short)
+  $lines = file($conf['changelog']);
+
+  // handle lines
+  for($i = count($lines)-1; $i >= 0; $i--){
+    $rec = _handleRecent($lines[$i], $ns, $flags);
+    if($rec !== false) {
+      if(--$first >= 0) continue; // skip first entries
+      $recent[] = $rec;
+      $count++;
+      // break when we have enough entries
+      if($count >= $num){ break; }
     }
+  }
 
-    fseek($fh,$pos);
-
-    $buf = fread($fh,$csz).$buf;            // prepend to buffer
-
-    $lines = explode("\n",$buf);            // split buffer into lines
-
-    if($pos > 0){
-      $buf = array_shift($lines);           // first one may be still incomplete
-    }
-
-    $cnt = count($lines);
-    if(!$cnt) continue;                     // no lines yet
-
-    // handle lines
-    for($i = $cnt-1; $i >= 0; $i--){
-      $rec = _handleRecent($lines[$i],$ns,$flags);
-      if($rec !== false){
-        if(--$first >= 0) continue;         // skip first entries
-        $recent[] = $rec;
-        $count++;
-
-        // break while when we have enough entries
-        if($count >= $num){
-          $pos = 0; // will break the while loop
-          break;    // will break the for loop
-        }
-      }
-    }
-  }// end of while
-
-  fclose($fh);
   return $recent;
 }
 
 /**
- * Compare the logline $a to the timestamp $b
- * @author Yann Hamon <yann.hamon@mandragor.org>
- * @return integer 0 if the logline has timestamp $b, <0 if the timestam
- *         of $a is greater than $b, >0 else.
- */
-function hasTimestamp($a, $b)
-{
-  if (strpos($a, $b) === 0)
-    return 0;
-  else
-    return strcmp ($a, $b);
-}
-
-/**
- * performs a dichotomic search on an array using
- * a custom compare function
+ * parses a changelog line into it's components
  *
- * @author Yann Hamon <yann.hamon@mandragor.org>
+ * @author Ben Coburn <btcoburn@silicodon.net>
  */
-function array_dichotomic_search($ar, $value, $compareFunc) {
-  $value = trim($value);
-  if (!$ar || !$value || !$compareFunc) return (null);
-  $len = count($ar);
-
-  $l = 0;
-  $r = $len-1;
-
-  do {
-    $i = floor(($l+$r)/2);
-    if ($compareFunc($ar[$i], $value)<0)
-      $l = $i+1;
-    else
-     $r = $i-1;
-  } while ($compareFunc($ar[$i], $value)!=0 && $l<=$r);
-
-  if ($compareFunc($ar[$i], $value)==0)
-    return $i;
-  else
-    return -1;
+function parseChangelogLine($line) {
+  $tmp = explode("\t", $line);
+    if ($tmp!==false && count($tmp)>1) {
+      $info = array();
+      $info['date']  = $tmp[0]; // unix timestamp
+      $info['ip']    = $tmp[1]; // IPv4 address (127.0.0.1)
+      $info['type']  = $tmp[2]; // log line type
+      $info['id']    = $tmp[3]; // page id
+      $info['user']  = $tmp[4]; // user name
+      $info['sum']   = $tmp[5]; // edit summary (or action reason)
+      $info['extra'] = rtrim($tmp[6], "\n"); // extra data (varies by line type)
+      return $info;
+  } else { return false; }
 }
 
 /**
- * gets additonal informations for a certain pagerevison
- * from the changelog
+ * Get the changelog information for a specific page id
+ * and revision (timestamp). Adjacent changelog lines
+ * are optimistically parsed and cached to speed up
+ * consecutive calls to getRevisionInfo. For large
+ * changelog files, only the chunk containing the
+ * requested changelog line is read.
  *
- * @author Andreas Gohr <andi@splitbrain.org>
- * @author Yann Hamon <yann.hamon@mandragor.org>
  * @author Ben Coburn <btcoburn@silicodon.net>
  */
-function getRevisionInfo($id,$rev,$mem_cache=true){
-  global $conf;
-  global $doku_temporary_revinfo_cache;
-  $cache =& $doku_temporary_revinfo_cache;
-  if(!$rev) return(null);
+function getRevisionInfo($id, $rev, $chunk_size=8192) {
+  global $cache_revinfo;
+  $cache =& $cache_revinfo;
+  if (!isset($cache[$id])) { $cache[$id] = array(); }
+  $rev = max($rev, 0);
 
   // check if it's already in the memory cache
-  if (is_array($cache) && isset($cache[$id]) && isset($cache[$id][$rev])) {
+  if (isset($cache[$id]) && isset($cache[$id][$rev])) {
     return $cache[$id][$rev];
   }
 
-  $info = array();
-  if(!@is_readable($conf['changelog'])){
-    msg($conf['changelog'].' is not readable',-1);
-    return $recent;
-  }
-  $loglines = file($conf['changelog']);
-
-  if (!$mem_cache) {
-    // Search for a line with a matching timestamp
-    $index = array_dichotomic_search($loglines, $rev, 'hasTimestamp');
-    if ($index == -1)
-      return;
-
-    // The following code is necessary when there is more than
-    // one line with one same timestamp
-    $loglines_matching = array();
-    for ($i=$index-1;$i>=0 && hasTimestamp($loglines[$i], $rev) == 0; $i--)
-      $loglines_matching[] = $loglines[$i];
-    $loglines_matching = array_reverse($loglines_matching);
-    $loglines_matching[] =  $loglines[$index];
-    $logsize = count($loglines);
-    for ($i=$index+1;$i<$logsize && hasTimestamp($loglines[$i], $rev) == 0; $i++)
-      $loglines_matching[] = $loglines[$i];
-
-    // pull off the line most recent line with the right id
-    $loglines_matching = array_reverse($loglines_matching); //newest first
-    foreach ($loglines_matching as $logline) {
-      $line = explode("\t", $logline);
-      if ($line[2]==$id) {
-        $info['date']  = $line[0];
-        $info['ip']    = $line[1];
-        $info['user']  = $line[3];
-        $info['sum']   = $line[4];
-        $info['minor'] = isMinor($info['sum']);
-        break;
+  $file = metaFN($id, '.changes');
+  if (!file_exists($file)) { return false; }
+  if (filesize($file)<$chunk_size || $chunk_size==0) {
+    // read whole file
+    $lines = file($file);
+    if ($lines===false) { return false; }
+  } else {
+    // read by chunk
+    $fp = fopen($file, 'rb'); // "file pointer"
+    if ($fp===false) { return false; }
+    $head = 0;
+    fseek($fp, 0, SEEK_END);
+    $tail = ftell($fp);
+    $finger = 0;
+    $finger_rev = 0;
+
+    // find chunk
+    while ($tail-$head>$chunk_size) {
+      $finger = $head+floor(($tail-$head)/2.0);
+      fseek($fp, $finger);
+      fgets($fp); // slip the finger forward to a new line
+      $finger = ftell($fp);
+      $tmp = fgets($fp); // then read at that location
+      $tmp = parseChangelogLine($tmp);
+      $finger_rev = $tmp['date'];
+      if ($finger==$head || $finger==$tail) { break; }
+      if ($finger_rev>$rev) {
+        $tail = $finger;
+      } else {
+        $head = $finger;
       }
     }
+
+    if ($tail-$head<1) {
+      // cound not find chunk, assume requested rev is missing
+      fclose($fp);
+      return false;
+    }
+
+    // read chunk
+    $chunk = '';
+    $chunk_size = max($tail-$head, 0); // found chunk size
+    $got = 0;
+    fseek($fp, $head);
+    while ($got<$chunk_size && !feof($fp)) {
+      $tmp = fread($fp, max($chunk_size-$got, 0));
+      if ($tmp===false) { break; } //error state
+      $got += strlen($tmp);
+      $chunk .= $tmp;
+    }
+    $lines = explode("\n", $chunk);
+    array_pop($lines); // remove trailing newline
+    fclose($fp);
+  }
+
+  // parse and cache changelog lines
+  foreach ($lines as $value) {
+    $tmp = parseChangelogLine($value);
+    if ($tmp!==false) {
+      $cache[$id][$tmp['date']] = $tmp;
+    }
+  }
+  if (!isset($cache[$id][$rev])) { return false; }
+  return $cache[$id][$rev];
+}
+
+/**
+ * Return a list of page revisions numbers
+ * Does not guarantee that the revision exists in the attic,
+ * only that a line with the date exists in the changelog.
+ * By default the current revision is skipped.
+ *
+ * id:    the page of interest
+ * first: skip the first n changelog lines
+ * num:   number of revisions to return
+ *
+ * The current revision is automatically skipped when the page exists.
+ * See $INFO['meta']['last_change'] for the current revision.
+ *
+ * For efficiency, the log lines are parsed and cached for later
+ * calls to getRevisionInfo. Large changelog files are read
+ * backwards in chunks untill the requested number of changelog
+ * lines are recieved.
+ *
+ * @author Ben Coburn <btcoburn@silicodon.net>
+ */
+function getRevisions($id, $first, $num, $chunk_size=8192) {
+  global $cache_revinfo;
+  $cache =& $cache_revinfo;
+  if (!isset($cache[$id])) { $cache[$id] = array(); }
+
+  $revs = array();
+  $lines = array();
+  $count  = 0;
+  $file = metaFN($id, '.changes');
+  $num = max($num, 0);
+  $chunk_size = max($chunk_size, 0);
+  if ($first<0) { $first = 0; }
+  else if (file_exists(wikiFN($id))) {
+     // skip current revision if the page exists
+    $first = max($first+1, 0);
+  }
+
+  if (!file_exists($file)) { return $revs; }
+  if (filesize($file)<$chunk_size || $chunk_size==0) {
+    // read whole file
+    $lines = file($file);
+    if ($lines===false) { return $revs; }
   } else {
-    // load and cache all the lines with the right id
-    if(!is_array($cache)) { $cache = array(); }
-    if (!isset($cache[$id])) { $cache[$id] = array(); }
-    foreach ($loglines as $logline) {
-      $start = strpos($logline, "\t", strpos($logline, "\t")+1)+1;
-      $end = strpos($logline, "\t", $start);
-      if (substr($logline, $start, $end-$start)==$id) {
-        $line = explode("\t", $logline);
-        $info = array();
-        $info['date']  = $line[0];
-        $info['ip']    = $line[1];
-        $info['user']  = $line[3];
-        $info['sum']   = $line[4];
-        $info['minor'] = isMinor($info['sum']);
-        $cache[$id][$info['date']] = $info;
+    // read chunks backwards
+    $fp = fopen($file, 'rb'); // "file pointer"
+    if ($fp===false) { return $revs; }
+    fseek($fp, 0, SEEK_END);
+    $tail = ftell($fp);
+
+    // chunk backwards
+    $finger = max($tail-$chunk_size, 0);
+    while ($count<$num+$first) {
+      fseek($fp, $finger);
+      if ($finger>0) {
+        fgets($fp); // slip the finger forward to a new line
+        $finger = ftell($fp);
+      }
+
+      // read chunk
+      if ($tail<=$finger) { break; }
+      $chunk = '';
+      $read_size = max($tail-$finger, 0); // found chunk size
+      $got = 0;
+      while ($got<$read_size && !feof($fp)) {
+        $tmp = fread($fp, max($read_size-$got, 0));
+        if ($tmp===false) { break; } //error state
+        $got += strlen($tmp);
+        $chunk .= $tmp;
+      }
+      $tmp = explode("\n", $chunk);
+      array_pop($tmp); // remove trailing newline
+
+      // combine with previous chunk
+      $count += count($tmp);
+      $lines = array_merge($tmp, $lines);
+
+      // next chunk
+      if ($finger==0) { break; } // already read all the lines
+      else {
+        $tail = $finger;
+        $finger = max($tail-$chunk_size, 0);
       }
     }
-    $info = $cache[$id][$rev];
+    fclose($fp);
   }
 
-  return $info;
-}
+  // skip parsing extra lines
+  $num = max(min(count($lines)-$first, $num), 0);
+  if      ($first>0 && $num>0)  { $lines = array_slice($lines, max(count($lines)-$first-$num, 0), $num); }
+  else if ($first>0 && $num==0) { $lines = array_slice($lines, 0, max(count($lines)-$first, 0)); }
+  else if ($first==0 && $num>0) { $lines = array_slice($lines, max(count($lines)-$num, 0)); }
 
+  // handle lines in reverse order
+  for ($i = count($lines)-1; $i >= 0; $i--) {
+    $tmp = parseChangelogLine($lines[$i]);
+    if ($tmp!==false) {
+      $cache[$id][$tmp['date']] = $tmp;
+      $revs[] = $tmp['date'];
+    }
+  }
+
+  return $revs;
+}
 
 /**
  * Saves a wikitext by calling io_writeWikiPage
  *
  * @author Andreas Gohr <andi@splitbrain.org>
+ * @author Ben Coburn <btcoburn@silicodon.net>
  */
 function saveWikiText($id,$text,$summary,$minor=false){
   global $conf;
   global $lang;
+  global $REV;
   // ignore if no changes were made
   if($text == rawWiki($id,'')){
     return;
@@ -1033,14 +1080,19 @@ function saveWikiText($id,$text,$summary,$minor=false){
 
   $file = wikiFN($id);
   $old  = saveOldRevision($id);
+  $wasRemoved = empty($text);
+  $wasCreated = !file_exists($file);
+  $wasReverted = ($REV==true);
 
-  if (empty($text)){
+  if ($wasRemoved){
     // remove empty file
     @unlink($file);
-    // remove any meta info
+    // remove old meta info...
     $mfiles = metaFiles($id);
+    $changelog = metaFN($id, '.changes');
     foreach ($mfiles as $mfile) {
-      if (file_exists($mfile)) @unlink($mfile);
+      // but keep per-page changelog to preserve page history
+      if (file_exists($mfile) && $mfile!==$changelog) { @unlink($mfile); }
     }
     $del = true;
     // autoset summary on deletion
@@ -1051,11 +1103,21 @@ function saveWikiText($id,$text,$summary,$minor=false){
   }else{
     // save file (namespace dir is created in io_writeWikiPage)
     io_writeWikiPage($file, $text, $id);
-    saveMetadata($id, $file, $minor);
     $del = false;
   }
 
-  addLogEntry(@filemtime($file),$id,$summary,$minor);
+  // select changelog line type
+  $extra = '';
+  $type = 'E';
+  if ($wasReverted) {
+    $type = 'R';
+    $extra = $REV;
+  }
+  else if ($wasCreated) { $type = 'C'; }
+  else if ($wasRemoved) { $type = 'D'; }
+  else if ($minor && $conf['useacl'] && $_SERVER['REMOTE_USER']) { $type = 'e'; } //minor edits only for logged in users
+
+  addLogEntry(@filemtime($file), $id, $type, $summary, $extra);
   // send notify mails
   notify($id,'admin',$old,$summary,$minor);
   notify($id,'subscribers',$old,$summary,$minor);
@@ -1066,27 +1128,6 @@ function saveWikiText($id,$text,$summary,$minor=false){
   }
 }
 
-/**
- * saves the metadata for a page
- *
- * @author Esther Brunner <wikidesign@gmail.com>
- */
-function saveMetadata($id, $file, $minor){
-  global $INFO;
-  
-  $user = $_SERVER['REMOTE_USER'];
-  
-  $meta = array();
-  if (!$INFO['exists']){ // newly created
-    $meta['date']['created'] = @filectime($file);
-    if ($user) $meta['creator'] = $INFO['userinfo']['name'];
-  } elseif (!$minor) {   // non-minor modification
-    $meta['date']['modified'] = @filemtime($file);
-    if ($user) $meta['contributor'][$user] = $INFO['userinfo']['name'];
-  }
-  p_set_metadata($id, $meta, true);
-}
-
 /**
  * moves the current version to the attic and returns its
  * revision date
@@ -1177,39 +1218,6 @@ function notify($id,$who,$rev='',$summary='',$minor=false,$replace=array()){
   mail_send($to,$subject,$text,$conf['mailfrom'],'',$bcc);
 }
 
-/**
- * Return a list of available page revisons
- *
- * @author Andreas Gohr <andi@splitbrain.org>
- */
-function getRevisions($id){
-  global $conf;
-
-  $id   = cleanID($id);
-  $revd = dirname(wikiFN($id,'foo'));
-  $id   = noNS($id);
-  $id   = utf8_encodeFN($id);
-  $len  = strlen($id);
-  $xlen = 10; // length of timestamp, strlen(time()) would be more correct, 
-              // but i don't expect dokuwiki still running in 287 years ;)
-              // so this will perform better
-
-  $revs = array();
-  if (is_dir($revd) && $dh = opendir($revd)) {
-    while (($file = readdir($dh)) !== false) {
-      if (substr($file,0,$len) === $id) {
-        $time = substr($file,$len+1,$xlen);
-        $time = str_replace('.','FOO',$time); // make sure a dot will make the next test fail
-        $time = (int) $time;
-        if($time) $revs[] = $time;
-      }
-    }
-    closedir($dh);
-  }
-  rsort($revs);
-  return $revs;
-}
-
 /**
  * extracts the query from a google referer
  *
@@ -1339,7 +1347,21 @@ function check(){
   if(is_writable($conf['changelog'])){
     msg('Changelog is writable',1);
   }else{
-    msg('Changelog is not writable',-1);
+    if (file_exists($conf['changelog'])) {
+      msg('Changelog is not writable',-1);
+    }
+  }
+
+  if (isset($conf['changelog_old']) && file_exists($conf['changelog_old'])) {
+    msg('Old changelog exists.', 0);
+  }
+
+  if (file_exists($conf['changelog'].'_failed')) {
+    msg('Importing old changelog failed.', -1);
+  } else if (file_exists($conf['changelog'].'_importing')) {
+    msg('Importing old changelog now.', 0);
+  } else if (file_exists($conf['changelog'].'_import_ok')) {
+    msg('Old changelog imported.', 1);
   }
 
   if(is_writable($conf['datadir'])){
diff --git a/inc/html.php b/inc/html.php
index 128cdeb009e7a586d43331d7ded8dc6e2720cbc9..4a56072e1232bb38882081bd69911b1fc3a719d7 100644
--- a/inc/html.php
+++ b/inc/html.php
@@ -442,19 +442,34 @@ function html_locked(){
  * list old revisions
  *
  * @author Andreas Gohr <andi@splitbrain.org>
+ * @author Ben Coburn <btcoburn@silicodon.net>
  */
-function html_revisions(){
+function html_revisions($first=0){
   global $ID;
   global $INFO;
   global $conf;
   global $lang;
-  $revisions = getRevisions($ID);
+  /* we need to get one additionally log entry to be able to
+   * decide if this is the last page or is there another one.
+   * see html_recent()
+   */
+  $revisions = getRevisions($ID, $first, $conf['recent']+1);
+  if(count($revisions)==0 && $first!=0){
+    $first=0;
+    $revisions = getRevisions($ID, $first, $conf['recent']+1);;
+  }
+  $hasNext = false;
+  if (count($revisions)>$conf['recent']) {
+    $hasNext = true;
+    array_pop($revisions); // remove extra log entry
+  }
+
   $date = @date($conf['dformat'],$INFO['lastmod']);
 
   print p_locale_xhtml('revisions');
   print '<ul>';
-  if($INFO['exists']){
-    print ($INFO['minor']) ? '<li class="minor">' : '<li>';
+  if($INFO['exists'] && $first==0){
+    print (isset($INFO['meta']) && isset($INFO['meta']['last_change']) && $INFO['meta']['last_change']['type']==='e') ? '<li class="minor">' : '<li>';
     print '<div class="li">';
 
     print $date;
@@ -477,7 +492,7 @@ function html_revisions(){
     $date = date($conf['dformat'],$rev);
     $info = getRevisionInfo($ID,$rev,true);
 
-    print ($info['minor']) ? '<li class="minor">' : '<li>';
+    print ($info['type']==='e') ? '<li class="minor">' : '<li>';
     print '<div class="li">';
     print $date;
 
@@ -507,6 +522,23 @@ function html_revisions(){
     print '</li>';
   }
   print '</ul>';
+
+  print '<div class="pagenav">';
+  $last = $first + $conf['recent'];
+  if ($first > 0) {
+    $first -= $conf['recent'];
+    if ($first < 0) $first = 0;
+    print '<div class="pagenav-prev">';
+    print html_btn('newer','',"p",array('do' => 'revisions', 'first' => $first));
+    print '</div>';
+  }
+  if ($hasNext) {
+    print '<div class="pagenav-next">';
+    print html_btn('older','',"n",array('do' => 'revisions', 'first' => $last));
+    print '</div>';
+  }
+  print '</div>';
+
 }
 
 /**
@@ -514,6 +546,7 @@ function html_revisions(){
  *
  * @author Andreas Gohr <andi@splitbrain.org>
  * @author Matthias Grimm <matthiasgrimm@users.sourceforge.net>
+ * @author Ben Coburn <btcoburn@silicodon.net>
  */
 function html_recent($first=0){
   global $conf;
@@ -526,16 +559,20 @@ function html_recent($first=0){
   $recents = getRecents($first,$conf['recent'] + 1,getNS($ID));
   if(count($recents) == 0 && $first != 0){
     $first=0;
-    $recents = getRecents(0,$conf['recent'] + 1,getNS($ID));
+    $recents = getRecents($first,$conf['recent'] + 1,getNS($ID));
+  }
+  $hasNext = false;
+  if (count($recents)>$conf['recent']) {
+    $hasNext = true;
+    array_pop($recents); // remove extra log entry
   }
-  $cnt = count($recents) <= $conf['recent'] ? count($recents) : $conf['recent'];
 
   print p_locale_xhtml('recent');
   print '<ul>';
 
   foreach($recents as $recent){
     $date = date($conf['dformat'],$recent['date']);
-    print ($recent['minor']) ? '<li class="minor">' : '<li>';
+    print ($recent['type']==='e') ? '<li class="minor">' : '<li>';
     print '<div class="li">';
 
     print $date.' ';
@@ -587,7 +624,7 @@ function html_recent($first=0){
     print html_btn('newer','',"p",array('do' => 'recent', 'first' => $first));
     print '</div>';
   }
-  if ($conf['recent'] < count($recents)) {
+  if ($hasNext) {
     print '<div class="pagenav-next">';
     print html_btn('older','',"n",array('do' => 'recent', 'first' => $last));
     print '</div>';
@@ -782,7 +819,7 @@ function html_diff($text='',$intro=true){
       $r = $REV;
     }else{
       //use last revision if none given
-      $revs = getRevisions($ID);
+      $revs = getRevisions($ID, 0, 1);
       $r = $revs[0];
     }
 
diff --git a/inc/init.php b/inc/init.php
index 01d2f74697774ca6ae48ea7a6d7531536fa50ba9..bfa22e0016b7e3b1ff4cbf5a4d16531087981f9e 100644
--- a/inc/init.php
+++ b/inc/init.php
@@ -24,8 +24,8 @@
   else { error_reporting(DOKU_E_LEVEL); }
 
   // init memory caches
-  global $cache_wikifn; $cache_wikifn = array();
-  global $cache_wikifn; $cache_cleanid = array();
+  $cache_wikifn = array();
+  $cache_cleanid = array();
 
   //prepare config array()
   global $conf;
@@ -128,8 +128,7 @@ function init_paths(){
                  'mediadir'  => 'media',
                  'metadir'   => 'meta',
                  'cachedir'  => 'cache',
-                 'lockdir'   => 'locks',
-                 'changelog' => 'changes.log');
+                 'lockdir'   => 'locks');
 
   foreach($paths as $c => $p){
     if(!$conf[$c])   $conf[$c] = $conf['savedir'].'/'.$p;
@@ -139,6 +138,12 @@ function init_paths(){
                                Or maybe you want to <a href=\"install.php\">run the
                                installer</a>?");
   }
+
+  // path to old changelog only needed for upgrading
+  $conf['changelog_old'] = init_path((isset($conf['changelog']))?($conf['changelog']):($conf['savedir'].'/changes.log'));
+  if ($conf['changelog_old']=='') { unset($conf['changelog_old']); }
+  // hardcoded changelog because it is now a cache that lives in meta
+  $conf['changelog'] = $conf['metadir'].'/_dokuwiki.changes';
 }
 
 /**
diff --git a/inc/template.php b/inc/template.php
index f065030430c573d77493317e8cbd4a16c7fde8f7..d30500e94b86c7bd01169dac7bcd4e9436cc841f 100644
--- a/inc/template.php
+++ b/inc/template.php
@@ -81,7 +81,8 @@ function tpl_content_core(){
       html_search();
       break;
     case 'revisions':
-      html_revisions();
+      $first = is_numeric($_REQUEST['first']) ? intval($_REQUEST['first']) : 0;
+      html_revisions($first);
       break;
     case 'diff':
       html_diff();
diff --git a/lib/exe/indexer.php b/lib/exe/indexer.php
index 2728e56656424e3662cc35c45021a1850ba3eec7..d6570791190bc5a4ce284927e8ae4d3134ee5553 100644
--- a/lib/exe/indexer.php
+++ b/lib/exe/indexer.php
@@ -27,7 +27,7 @@ if(@ignore_user_abort()){
 if(!$_REQUEST['debug']) ob_start();
 
 // run one of the jobs
-runIndexer() or metaUpdate() or runSitemapper();
+runIndexer() or metaUpdate() or runSitemapper() or runTrimRecentChanges();
 if($defer) sendGIF();
 
 if(!$_REQUEST['debug']) ob_end_clean();
@@ -35,6 +35,73 @@ exit;
 
 // --------------------------------------------------------------------
 
+/**
+ * Trims the recent changes cache (or imports the old changelog) as needed.
+ *
+ * @author Ben Coburn <btcoburn@silicodon.net>
+ */
+function runTrimRecentChanges() {
+    global $conf;
+
+    // Import old changelog (if needed)
+    // Uses the imporoldchangelog plugin to upgrade the changelog automaticaly.
+    // FIXME: Remove this from runTrimRecentChanges when it is no longer needed.
+    if (isset($conf['changelog_old']) &&
+        file_exists($conf['changelog_old']) && !file_exists($conf['changelog']) &&
+        !file_exists($conf['changelog'].'_importing') && !file_exists($conf['changelog'].'_tmp')) {
+            $tmp = array(); // no event data
+            trigger_event('TEMPORARY_CHANGELOG_UPGRADE_EVENT', $tmp);
+            return true;
+    }
+
+    // Trim the Recent Changes
+    // Trims the recent changes cache to the last $conf['changes_days'] recent
+    // changes or $conf['recent'] items, which ever is larger.
+    // The trimming is only done once a day.
+    if (file_exists($conf['changelog']) &&
+        (filectime($conf['changelog'])+86400)<time() &&
+        !file_exists($conf['changelog'].'_tmp')) {
+            io_lock($conf['changelog']);
+            $lines = file($conf['changelog']);
+            if (count($lines)<$conf['recent']) {
+                // nothing to trim
+                io_unlock($conf['changelog']);
+                return true;
+            }
+            // trim changelog
+            io_saveFile($conf['changelog'].'_tmp', ''); // presave tmp as 2nd lock
+            $kept = 0;
+            $trim_time = time() - $conf['recent_days']*86400;
+            $out_lines = array();
+            // check lines from newest to oldest
+            for ($i = count($lines)-1; $i >= 0; $i--) {
+                $tmp = parseChangelogLine($lines[$i]);
+                if ($tmp===false) { continue; }
+                if ($tmp['date']>$trim_time || $kept<$conf['recent']) {
+                    array_push($out_lines, implode("\t", $tmp)."\n");
+                    $kept++;
+                } else {
+                    // no more lines worth keeping
+                    break;
+                }
+            }
+            io_saveFile($conf['changelog'].'_tmp', implode('', $out_lines));
+            unlink($conf['changelog']);
+            if (!rename($conf['changelog'].'_tmp', $conf['changelog'])) {
+                // rename failed so try another way...
+                io_unlock($conf['changelog']);
+                io_saveFile($conf['changelog'], implode('', $out_lines));
+                unlink($conf['changelog'].'_tmp');
+            } else {
+                io_unlock($conf['changelog']);
+            }
+            return true;
+    }
+
+    // nothing done
+    return false;
+}
+
 /**
  * Runs the indexer for the current page
  *
diff --git a/lib/plugins/config/lang/en/lang.php b/lib/plugins/config/lang/en/lang.php
index c2bd5aacfd91776d2f839a3ccb3966118b2ba845..4c4c713e56a24c2293716b5cc2d2981797cd9007 100644
--- a/lib/plugins/config/lang/en/lang.php
+++ b/lib/plugins/config/lang/en/lang.php
@@ -124,6 +124,7 @@ $lang['sitemap']     = 'Generate Google sitemap (days)';
 $lang['rss_type']    = 'XML feed type';
 $lang['rss_linkto']  = 'XML feed links to';
 $lang['rss_update']  = 'XML feed update interval (sec)';
+$lang['recent_days'] = 'How many recent changes to keep (days)';
 
 /* Target options */
 $lang['target____wiki']      = 'Target window for internal links';
diff --git a/lib/plugins/config/settings/config.metadata.php b/lib/plugins/config/settings/config.metadata.php
index 0dd9f1de30d8bf86dd89184b9b8066a84c48f61f..b55c0e930f5aecf784e283d21a0e03fdd8ab6e03 100644
--- a/lib/plugins/config/settings/config.metadata.php
+++ b/lib/plugins/config/settings/config.metadata.php
@@ -162,6 +162,7 @@ $meta['sitemap']     = array('numeric');
 $meta['rss_type']    = array('multichoice','_choices' => array('rss','rss1','rss2','atom'));
 $meta['rss_linkto']  = array('multichoice','_choices' => array('diff','page','rev','current'));
 $meta['rss_update']  = array('numeric');
+$meta['recent_days'] = array('numeric');
 
 $meta['_network']    = array('fieldset');
 $meta['proxy____host'] = array('string','_pattern' => '#^[a-z0-9\-\.+]+?#i');
diff --git a/lib/plugins/importoldchangelog/action.php b/lib/plugins/importoldchangelog/action.php
new file mode 100644
index 0000000000000000000000000000000000000000..400ff6a183986962c5645a650b1434fdcc1babe4
--- /dev/null
+++ b/lib/plugins/importoldchangelog/action.php
@@ -0,0 +1,177 @@
+<?php
+// must be run within Dokuwiki
+if(!defined('DOKU_INC')) die();
+
+if(!defined('DOKU_PLUGIN')) define('DOKU_PLUGIN',DOKU_INC.'lib/plugins/');
+require_once(DOKU_PLUGIN.'action.php');
+
+class action_plugin_importoldchangelog extends DokuWiki_Action_Plugin {
+
+	function getInfo(){
+		return array(
+			'author' => 'Ben Coburn',
+			'email'  => 'btcoburn@silicodon.net',
+			'date'   => '2006-08-30',
+			'name'   => 'Import Old Changelog',
+			'desc'   => 'Imports and converts the single file changelog '.
+                        'from the 2006-03-09b release to the new format. '.
+                        'Also reconstructs missing changelog data from  '.
+                        'old revisions kept in the attic.',
+            'url'    => 'http://wiki.splitbrain.org/wiki:changelog'
+			);
+	}
+
+	function register(&$controller) {
+        $controller->register_hook('TEMPORARY_CHANGELOG_UPGRADE_EVENT', 'BEFORE', $this, 'run_import');
+	}
+
+    function importOldLog($line, &$logs) {
+        global $lang;
+        /*
+        // Note: old log line format
+        //$info['date']  = $tmp[0];
+        //$info['ip']    = $tmp[1];
+        //$info['id']    = $tmp[2];
+        //$info['user']  = $tmp[3];
+        //$info['sum']   = $tmp[4];
+        */
+        $oldline = @explode("\t", $line);
+        if ($oldline!==false && count($oldline)>1) {
+            // trim summary
+            $wasMinor = (substr($oldline[4], 0, 1)==='*');
+            $sum = rtrim(substr($oldline[4], 1), "\n");
+            // guess line type
+            $type = 'E';
+            if ($wasMinor) { $type = 'e'; }
+            if ($sum===$lang['created']) { $type = 'C'; }
+            if ($sum===$lang['deleted']) { $type = 'D'; }
+            // build new log line
+            $tmp = array();
+            $tmp['date']  = $oldline[0];
+            $tmp['ip']    = $oldline[1];
+            $tmp['type']  = $type;
+            $tmp['id']    = $oldline[2];
+            $tmp['user']  = $oldline[3];
+            $tmp['sum']   = $sum;
+            $tmp['extra'] = '';
+            // order line by id
+            if (!isset($logs[$tmp['id']])) { $logs[$tmp['id']] = array(); }
+            $logs[$tmp['id']][$tmp['date']] = $tmp;
+        }
+    }
+
+    function importFromAttic(&$logs) {
+        global $conf, $lang;
+        $base = $conf['olddir'];
+        $stack = array('');
+        $context = ''; // namespace
+        while (count($stack)>0){
+            $context = array_pop($stack);
+            $dir = dir($base.'/'.str_replace(':', '/', $context));
+
+            while (($file = $dir->read()) !== false) {
+                if ($file==='.' || $file==='..') { continue; }
+                $matches = array();
+                if (preg_match('/([^.]*)\.([^.]*)\..*/', $file, $matches)===1) {
+                    $id = (($context=='')?'':$context.':').$matches[1];
+                    $date = $matches[2];
+
+                    // check if page & revision are already logged
+                    if (!isset($logs[$id])) { $logs[$id] = array(); }
+                    if (!isset($logs[$id][$date])) {
+                        $tmp = array();
+                        $tmp['date']  = $date;
+                        $tmp['ip']    = '127.0.0.1'; // original ip lost
+                        $tmp['type']  = 'E';
+                        $tmp['id']    = $id;
+                        $tmp['user']  = ''; // original user lost
+                        $tmp['sum']   = '('.$lang['restored'].')'; // original summary lost
+                        $tmp['extra'] = '';
+                        $logs[$id][$date] = $tmp;
+                    }
+
+                } else if (is_dir($dir->path.'/'.$file)) {
+                    array_push($stack, (($context=='')?'':$context.':').$file);
+                }
+
+            }
+
+            $dir->close();
+        }
+
+    }
+
+    function savePerPageChanges($id, &$changes, &$recent, $trim_time) {
+        $out_lines = array();
+        ksort($changes); // ensure correct order of changes from attic
+        foreach ($changes as $tmp) {
+            $line = implode("\t", $tmp)."\n";
+            array_push($out_lines, $line);
+            if ($tmp['date']>$trim_time) {
+                $recent[$tmp['date']] = $line;
+            }
+        }
+        io_saveFile(metaFN($id, '.changes'), implode('', $out_lines));
+    }
+
+    function resetTimer() {
+        // Add 5 minutes to the script execution timer...
+        // This should be much more than needed.
+        set_time_limit(5*60);
+        // Note: Has no effect in safe-mode!
+    }
+
+    function run_import(&$event, $args) {
+        global $conf;
+        register_shutdown_function('importoldchangelog_plugin_shutdown');
+        touch($conf['changelog'].'_importing'); // changelog importing lock
+        io_saveFile($conf['changelog'], ''); // pre-create changelog
+        io_lock($conf['changelog']);  // hold onto the lock
+        // load old changelog
+        $this->resetTimer();
+        $log = array();
+        $oldlog = file($conf['changelog_old']);
+        foreach ($oldlog as $line) {
+            $this->importOldLog($line, $log);
+        }
+        unset($oldlog); // free memory
+        // look in the attic for unlogged revisions
+        $this->resetTimer();
+        $this->importFromAttic($log);
+        // save per-page changelogs
+        $this->resetTimer();
+        $recent = array();
+        $trim_time = time() - $conf['recent_days']*86400;
+        foreach ($log as $id => $page) {
+            $this->savePerPageChanges($id, $page, $recent, $trim_time);
+        }
+        // save recent changes cache
+        $this->resetTimer();
+        ksort($recent); // ensure correct order of recent changes
+        io_unlock($conf['changelog']); // hand off the lock to io_saveFile
+        io_saveFile($conf['changelog'], implode('', $recent));
+        unlink($conf['changelog'].'_importing'); // changelog importing unlock
+    }
+
+}
+
+function importoldchangelog_plugin_shutdown() {
+    global $conf;
+    $path = array();
+    $path['changelog'] = $conf['changelog'];
+    $path['importing'] = $conf['changelog'].'_importing';
+    $path['failed']    = $conf['changelog'].'_failed';
+    $path['import_ok'] = $conf['changelog'].'_import_ok';
+    io_unlock($path['changelog']); // guarantee unlocking
+    if (file_exists($path['importing'])) {
+        // import did not finish
+        rename($path['importing'], $path['failed']) or trigger_error('Importing changelog failed.', E_USER_WARNING);
+        @unlink($path['import_ok']);
+    } else {
+        // import successful
+        touch($path['import_ok']);
+        @unlink($path['failed']);
+    }
+}
+
+
diff --git a/lib/plugins/plugin/admin.php b/lib/plugins/plugin/admin.php
index ef142c4cca94a4c6ac31417c008275ed7af65c38..2c47de665a2291066b07cd7131c8bd95b829c06a 100644
--- a/lib/plugins/plugin/admin.php
+++ b/lib/plugins/plugin/admin.php
@@ -23,7 +23,7 @@ require_once(DOKU_PLUGIN.'admin.php');
 		
 		// plugins that are an integral part of dokuwiki, they shouldn't be disabled or deleted
 		global $plugin_protected;
-		$plugin_protected = array('acl','plugin','config','info','usermanager');
+		$plugin_protected = array('acl','plugin','config','info','usermanager', 'importoldchangelog');
 		
 /**
  * All DokuWiki plugins to extend the admin function