1ed7b5f09Sandi<?php 215fae107Sandi/** 315fae107Sandi * Common DokuWiki functions 415fae107Sandi * 515fae107Sandi * @license GPL 2 (http://www.gnu.org/licenses/gpl.html) 615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 715fae107Sandi */ 815fae107Sandi 9ed7b5f09Sandi if(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../').'/'); 10ed7b5f09Sandi require_once(DOKU_INC.'conf/dokuwiki.php'); 11ed7b5f09Sandi require_once(DOKU_INC.'inc/io.php'); 12ed7b5f09Sandi require_once(DOKU_INC.'inc/utf8.php'); 13ed7b5f09Sandi require_once(DOKU_INC.'inc/mail.php'); 14f3f0262cSandi 15f3f0262cSandi/** 1615fae107Sandi * Return info about the current document as associative 17f3f0262cSandi * array. 1815fae107Sandi * 1915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 20f3f0262cSandi */ 21f3f0262cSandifunction pageinfo(){ 22f3f0262cSandi global $ID; 23f3f0262cSandi global $REV; 24f3f0262cSandi global $USERINFO; 25f3f0262cSandi global $conf; 26f3f0262cSandi 27f3f0262cSandi if($_SERVER['REMOTE_USER']){ 28f3f0262cSandi $info['user'] = $_SERVER['REMOTE_USER']; 29f3f0262cSandi $info['userinfo'] = $USERINFO; 30f3f0262cSandi $info['perm'] = auth_quickaclcheck($ID); 31f3f0262cSandi }else{ 32f3f0262cSandi $info['user'] = ''; 33f3f0262cSandi $info['perm'] = auth_aclcheck($ID,'',null); 34f3f0262cSandi } 35f3f0262cSandi 36f3f0262cSandi $info['namespace'] = getNS($ID); 37f3f0262cSandi $info['locked'] = checklock($ID); 38f3f0262cSandi $info['filepath'] = realpath(wikiFN($ID,$REV)); 39f3f0262cSandi $info['exists'] = @file_exists($info['filepath']); 40f3f0262cSandi if($REV && !$info['exists']){ 41f3f0262cSandi //check if current revision was meant 42f3f0262cSandi $cur = wikiFN($ID); 43f3f0262cSandi if(@file_exists($cur) && (@filemtime($cur) == $REV)){ 44f3f0262cSandi $info['filepath'] = realpath($cur); 45f3f0262cSandi $info['exists'] = true; 46f3f0262cSandi $REV = ''; 47f3f0262cSandi } 48f3f0262cSandi } 49f3f0262cSandi if($info['exists']){ 50f3f0262cSandi $info['writable'] = (is_writable($info['filepath']) && 51f3f0262cSandi ($info['perm'] >= AUTH_EDIT)); 52f3f0262cSandi }else{ 53f3f0262cSandi $info['writable'] = ($info['perm'] >= AUTH_CREATE); 54f3f0262cSandi } 55f3f0262cSandi $info['editable'] = ($info['writable'] && empty($info['lock'])); 56f3f0262cSandi $info['lastmod'] = @filemtime($info['filepath']); 57f3f0262cSandi 58652610a2Sandi //who's the editor 59652610a2Sandi if($REV){ 60652610a2Sandi $revinfo = getRevisionInfo($ID,$REV); 61652610a2Sandi }else{ 62652610a2Sandi $revinfo = getRevisionInfo($ID,$info['lastmod']); 63652610a2Sandi } 64652610a2Sandi $info['ip'] = $revinfo['ip']; 65652610a2Sandi $info['user'] = $revinfo['user']; 66652610a2Sandi $info['sum'] = $revinfo['sum']; 67652610a2Sandi $info['editor'] = $revinfo['ip']; 68652610a2Sandi if($revinfo['user']) $info['editor'].= ' ('.$revinfo['user'].')'; 69652610a2Sandi 70f3f0262cSandi return $info; 71f3f0262cSandi} 72f3f0262cSandi 73f3f0262cSandi/** 740396becbSandi * print a message 750396becbSandi * 760396becbSandi * If HTTP headers were not sent yet the message is added 770396becbSandi * to the global message array else it's printed directly 780396becbSandi * using html_msgarea() 790396becbSandi * 80f3f0262cSandi * 81f3f0262cSandi * Levels can be: 82f3f0262cSandi * 83f3f0262cSandi * -1 error 84f3f0262cSandi * 0 info 85f3f0262cSandi * 1 success 8615fae107Sandi * 8715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 880396becbSandi * @see html_msgarea 89f3f0262cSandi */ 90f3f0262cSandifunction msg($message,$lvl=0){ 91f3f0262cSandi global $MSG; 92f3f0262cSandi $errors[-1] = 'error'; 93f3f0262cSandi $errors[0] = 'info'; 94f3f0262cSandi $errors[1] = 'success'; 95f3f0262cSandi 96cc20ad51Sandi if(!headers_sent()){ 97f3f0262cSandi if(!isset($MSG)) $MSG = array(); 98f3f0262cSandi $MSG[]=array('lvl' => $errors[$lvl], 'msg' => $message); 990396becbSandi }else{ 1000396becbSandi $MSG = array(); 1010396becbSandi $MSG[]=array('lvl' => $errors[$lvl], 'msg' => $message); 1020396becbSandi html_msgarea(); 1030396becbSandi } 104f3f0262cSandi} 105f3f0262cSandi 106f3f0262cSandi/** 10715fae107Sandi * This builds the breadcrumb trail and returns it as array 10815fae107Sandi * 10915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 110f3f0262cSandi */ 111f3f0262cSandifunction breadcrumbs(){ 112f3f0262cSandi global $ID; 113f3f0262cSandi global $ACT; 114f3f0262cSandi global $conf; 115f3f0262cSandi $crumbs = $_SESSION[$conf['title']]['bc']; 116f3f0262cSandi 117f3f0262cSandi //first visit? 118f3f0262cSandi if (!is_array($crumbs)){ 119f3f0262cSandi $crumbs = array(); 120f3f0262cSandi } 121f3f0262cSandi //we only save on show and existing wiki documents 122f3f0262cSandi if($ACT != 'show' || !@file_exists(wikiFN($ID))){ 123f3f0262cSandi $_SESSION[$conf['title']]['bc'] = $crumbs; 124f3f0262cSandi return $crumbs; 125f3f0262cSandi } 126f3f0262cSandi //remove ID from array 127f3f0262cSandi $pos = array_search($ID,$crumbs); 128f3f0262cSandi if($pos !== false && $pos !== null){ 129f3f0262cSandi array_splice($crumbs,$pos,1); 130f3f0262cSandi } 131f3f0262cSandi 132f3f0262cSandi //add to array 133f3f0262cSandi $crumbs[] =$ID; 134f3f0262cSandi //reduce size 135f3f0262cSandi while(count($crumbs) > $conf['breadcrumbs']){ 136f3f0262cSandi array_shift($crumbs); 137f3f0262cSandi } 138f3f0262cSandi //save to session 139f3f0262cSandi $_SESSION[$conf['title']]['bc'] = $crumbs; 140f3f0262cSandi return $crumbs; 141f3f0262cSandi} 142f3f0262cSandi 143f3f0262cSandi/** 14415fae107Sandi * Filter for page IDs 14515fae107Sandi * 146f3f0262cSandi * This is run on a ID before it is outputted somewhere 147f3f0262cSandi * currently used to replace the colon with something else 148f3f0262cSandi * on Windows systems and to have proper URL encoding 14915fae107Sandi * 15049c713a3Sandi * Urlencoding is ommitted when the second parameter is false 15149c713a3Sandi * 15215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 153f3f0262cSandi */ 15449c713a3Sandifunction idfilter($id,$ue=true){ 155f3f0262cSandi global $conf; 156f3f0262cSandi if ($conf['useslash'] && $conf['userewrite']){ 157f3f0262cSandi $id = strtr($id,':','/'); 158f3f0262cSandi }elseif (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN' && 159f3f0262cSandi $conf['userewrite']) { 160f3f0262cSandi $id = strtr($id,':',';'); 161f3f0262cSandi } 16249c713a3Sandi if($ue){ 163f3f0262cSandi $id = urlencode($id); 164f3f0262cSandi $id = str_replace('%3A',':',$id); //keep as colon 165f3f0262cSandi $id = str_replace('%2F','/',$id); //keep as slash 16649c713a3Sandi } 167f3f0262cSandi return $id; 168f3f0262cSandi} 169f3f0262cSandi 170f3f0262cSandi/** 171ed7b5f09Sandi * This builds a link to a wikipage 17215fae107Sandi * 17315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 174f3f0262cSandi */ 175ed7b5f09Sandifunction wl($id='',$more='',$abs=false){ 176f3f0262cSandi global $conf; 177f3f0262cSandi $more = str_replace(',','&',$more); 178f3f0262cSandi 179f3f0262cSandi $id = idfilter($id); 180ed7b5f09Sandi if($abs){ 181ed7b5f09Sandi $xlink = DOKU_URL; 182ed7b5f09Sandi }else{ 183ed7b5f09Sandi $xlink = DOKU_BASE; 184ed7b5f09Sandi } 185f3f0262cSandi 186f3f0262cSandi if(!$conf['userewrite']){ 187ed7b5f09Sandi $xlink .= DOKU_SCRIPT.'?id='.$id; 188f3f0262cSandi if($more) $xlink .= '&'.$more; 189f3f0262cSandi }else{ 190f3f0262cSandi $xlink .= $id; 191f3f0262cSandi if($more) $xlink .= '?'.$more; 192f3f0262cSandi } 193f3f0262cSandi 194f3f0262cSandi return $xlink; 195f3f0262cSandi} 196f3f0262cSandi 197f3f0262cSandi/** 198f3f0262cSandi * Just builds a link to a script 19915fae107Sandi * 200ed7b5f09Sandi * @todo maybe obsolete 20115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 202f3f0262cSandi */ 203f3f0262cSandifunction script($script='doku.php'){ 204ed7b5f09Sandi# $link = getBaseURL(); 205ed7b5f09Sandi# $link .= $script; 206ed7b5f09Sandi# return $link; 207ed7b5f09Sandi return DOKU_BASE.DOKU_SCRIPT; 208f3f0262cSandi} 209f3f0262cSandi 210f3f0262cSandi/** 211f3f0262cSandi * Return namespacepart of a wiki ID 21215fae107Sandi * 21315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 214f3f0262cSandi */ 215f3f0262cSandifunction getNS($id){ 216f3f0262cSandi if(strpos($id,':')!==false){ 217f3f0262cSandi return substr($id,0,strrpos($id,':')); 218f3f0262cSandi } 219f3f0262cSandi return false; 220f3f0262cSandi} 221f3f0262cSandi 222f3f0262cSandi/** 22315fae107Sandi * Returns the ID without the namespace 22415fae107Sandi * 22515fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 226f3f0262cSandi */ 227f3f0262cSandifunction noNS($id){ 228f3f0262cSandi return preg_replace('/.*:/','',$id); 229f3f0262cSandi} 230f3f0262cSandi 231f3f0262cSandi/** 23215fae107Sandi * Spamcheck against wordlist 23315fae107Sandi * 234f3f0262cSandi * Checks the wikitext against a list of blocked expressions 235f3f0262cSandi * returns true if the text contains any bad words 23615fae107Sandi * 23715fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 238f3f0262cSandi */ 239f3f0262cSandifunction checkwordblock(){ 240f3f0262cSandi global $TEXT; 241f3f0262cSandi global $conf; 242f3f0262cSandi 243f3f0262cSandi if(!$conf['usewordblock']) return false; 244f3f0262cSandi 245703f6fdeSandi $blockfile = file('conf/wordblock.conf'); 2463e2965d7Sandi //how many lines to read at once (to work around some PCRE limits) 2473e2965d7Sandi if(version_compare(phpversion(),'4.3.0','<')){ 2483e2965d7Sandi //old versions of PCRE define a maximum of parenthesises even if no 2493e2965d7Sandi //backreferences are used - the maximum is 99 2503e2965d7Sandi //this is very bad performancewise and may even be too high still 2513e2965d7Sandi $chunksize = 40; 2523e2965d7Sandi }else{ 253703f6fdeSandi //read file in chunks of 600 - this should work around the 2543e2965d7Sandi //MAX_PATTERN_SIZE in modern PCRE 2553e2965d7Sandi $chunksize = 600; 2563e2965d7Sandi } 2573e2965d7Sandi while($blocks = array_splice($blockfile,0,$chunksize)){ 258f3f0262cSandi $re = array(); 259f3f0262cSandi #build regexp from blocks 260f3f0262cSandi foreach($blocks as $block){ 261f3f0262cSandi $block = preg_replace('/#.*$/','',$block); 262f3f0262cSandi $block = trim($block); 263f3f0262cSandi if(empty($block)) continue; 264f3f0262cSandi $re[] = $block; 265f3f0262cSandi } 266f3f0262cSandi if(preg_match('#('.join('|',$re).')#si',$TEXT)) return true; 267703f6fdeSandi } 268f3f0262cSandi return false; 269f3f0262cSandi} 270f3f0262cSandi 271f3f0262cSandi/** 27215fae107Sandi * Return the IP of the client 27315fae107Sandi * 27415fae107Sandi * Honours X-Forwarded-For Proxy Headers 27515fae107Sandi * 27615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 277f3f0262cSandi */ 278f3f0262cSandifunction clientIP(){ 279f3f0262cSandi $my = $_SERVER['REMOTE_ADDR']; 280f3f0262cSandi if($_SERVER['HTTP_X_FORWARDED_FOR']){ 281f3f0262cSandi $my .= ' ('.$_SERVER['HTTP_X_FORWARDED_FOR'].')'; 282f3f0262cSandi } 283f3f0262cSandi return $my; 284f3f0262cSandi} 285f3f0262cSandi 286f3f0262cSandi/** 28715fae107Sandi * Checks if a given page is currently locked. 28815fae107Sandi * 289f3f0262cSandi * removes stale lockfiles 29015fae107Sandi * 29115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 292f3f0262cSandi */ 293f3f0262cSandifunction checklock($id){ 294f3f0262cSandi global $conf; 295f3f0262cSandi $lock = wikiFN($id).'.lock'; 296f3f0262cSandi 297f3f0262cSandi //no lockfile 298f3f0262cSandi if(!@file_exists($lock)) return false; 299f3f0262cSandi 300f3f0262cSandi //lockfile expired 301f3f0262cSandi if((time() - filemtime($lock)) > $conf['locktime']){ 302f3f0262cSandi unlink($lock); 303f3f0262cSandi return false; 304f3f0262cSandi } 305f3f0262cSandi 306f3f0262cSandi //my own lock 307f3f0262cSandi $ip = io_readFile($lock); 308f3f0262cSandi if( ($ip == clientIP()) || ($ip == $_SERVER['REMOTE_USER']) ){ 309f3f0262cSandi return false; 310f3f0262cSandi } 311f3f0262cSandi 312f3f0262cSandi return $ip; 313f3f0262cSandi} 314f3f0262cSandi 315f3f0262cSandi/** 31615fae107Sandi * Lock a page for editing 31715fae107Sandi * 31815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 319f3f0262cSandi */ 320f3f0262cSandifunction lock($id){ 321f3f0262cSandi $lock = wikiFN($id).'.lock'; 322f3f0262cSandi if($_SERVER['REMOTE_USER']){ 323f3f0262cSandi io_saveFile($lock,$_SERVER['REMOTE_USER']); 324f3f0262cSandi }else{ 325f3f0262cSandi io_saveFile($lock,clientIP()); 326f3f0262cSandi } 327f3f0262cSandi} 328f3f0262cSandi 329f3f0262cSandi/** 33015fae107Sandi * Unlock a page if it was locked by the user 331f3f0262cSandi * 33215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 33315fae107Sandi * @return bool true if a lock was removed 334f3f0262cSandi */ 335f3f0262cSandifunction unlock($id){ 336f3f0262cSandi $lock = wikiFN($id).'.lock'; 337f3f0262cSandi if(@file_exists($lock)){ 338f3f0262cSandi $ip = io_readFile($lock); 339f3f0262cSandi if( ($ip == clientIP()) || ($ip == $_SERVER['REMOTE_USER']) ){ 340f3f0262cSandi @unlink($lock); 341f3f0262cSandi return true; 342f3f0262cSandi } 343f3f0262cSandi } 344f3f0262cSandi return false; 345f3f0262cSandi} 346f3f0262cSandi 347f3f0262cSandi/** 34815fae107Sandi * Remove unwanted chars from ID 34915fae107Sandi * 350f3f0262cSandi * Cleans a given ID to only use allowed characters. Accented characters are 351f3f0262cSandi * converted to unaccented ones 35215fae107Sandi * 35315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 354f3f0262cSandi */ 355f3f0262cSandifunction cleanID($id){ 356f3f0262cSandi global $conf; 357f3f0262cSandi global $lang; 358f3f0262cSandi $id = trim($id); 359c41c03f3Sandi $id = utf8_strtolower($id); 360f3f0262cSandi 361f3f0262cSandi //alternative namespace seperator 362f3f0262cSandi $id = strtr($id,';',':'); 3633021e063Sandi if($conf['useslash']){ 3643021e063Sandi $id = strtr($id,'/',':'); 3653021e063Sandi }else{ 3663021e063Sandi $id = strtr($id,'/','_'); 3673021e063Sandi } 368f3f0262cSandi 3698b709e9dSandi if($conf['deaccent']) $id = utf8_deaccent($id,-1); 370c41c03f3Sandi 371099ada41Sandi //remove specials 372099ada41Sandi $id = utf8_stripspecials($id,'_','_:.-'); 373c41c03f3Sandi 374c41c03f3Sandi //clean up 375f3f0262cSandi $id = preg_replace('#__#','_',$id); 376f3f0262cSandi $id = preg_replace('#:+#',':',$id); 377f3f0262cSandi $id = trim($id,':._-'); 378f3f0262cSandi $id = preg_replace('#:[:\._\-]+#',':',$id); 379f3f0262cSandi 380f3f0262cSandi return($id); 381f3f0262cSandi} 382f3f0262cSandi 383f3f0262cSandi/** 384f3f0262cSandi * returns the full path to the datafile specified by ID and 385f3f0262cSandi * optional revision 38615fae107Sandi * 38749c713a3Sandi * The filename is URL encoded to protect Unicode chars 38849c713a3Sandi * 38915fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 390f3f0262cSandi */ 391f3f0262cSandifunction wikiFN($id,$rev=''){ 392f3f0262cSandi global $conf; 393f3f0262cSandi $id = cleanID($id); 394f3f0262cSandi $id = str_replace(':','/',$id); 395f3f0262cSandi if(empty($rev)){ 39649c713a3Sandi $fn = $conf['datadir'].'/'.$id.'.txt'; 397f3f0262cSandi }else{ 398f3f0262cSandi $fn = $conf['olddir'].'/'.$id.'.'.$rev.'.txt'; 39949c713a3Sandi if($conf['usegzip'] && !@file_exists($fn)){ 40049c713a3Sandi //return gzip if enabled and plaintext doesn't exist 40149c713a3Sandi $fn .= '.gz'; 40249c713a3Sandi } 40349c713a3Sandi } 40449c713a3Sandi $fn = utf8_encodeFN($fn); 405f3f0262cSandi return $fn; 406f3f0262cSandi} 407f3f0262cSandi 408f3f0262cSandi/** 409f3f0262cSandi * Returns the full filepath to a localized textfile if local 410f3f0262cSandi * version isn't found the english one is returned 41115fae107Sandi * 41215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 413f3f0262cSandi */ 414f3f0262cSandifunction localeFN($id){ 415f3f0262cSandi global $conf; 416f3f0262cSandi $file = './lang/'.$conf['lang'].'/'.$id.'.txt'; 417f3f0262cSandi if(!@file_exists($file)){ 418f3f0262cSandi //fall back to english 419f3f0262cSandi $file = './lang/en/'.$id.'.txt'; 420f3f0262cSandi } 421f3f0262cSandi return cleanText($file); 422f3f0262cSandi} 423f3f0262cSandi 424f3f0262cSandi/** 425f3f0262cSandi * convert line ending to unix format 426f3f0262cSandi * 42715fae107Sandi * @see formText() for 2crlf conversion 42815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 429f3f0262cSandi */ 430f3f0262cSandifunction cleanText($text){ 431f3f0262cSandi $text = preg_replace("/(\015\012)|(\015)/","\012",$text); 432f3f0262cSandi return $text; 433f3f0262cSandi} 434f3f0262cSandi 435f3f0262cSandi/** 436f3f0262cSandi * Prepares text for print in Webforms by encoding special chars. 437f3f0262cSandi * It also converts line endings to Windows format which is 438f3f0262cSandi * pseudo standard for webforms. 439f3f0262cSandi * 44015fae107Sandi * @see cleanText() for 2unix conversion 44115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 442f3f0262cSandi */ 443f3f0262cSandifunction formText($text){ 444f3f0262cSandi $text = preg_replace("/\012/","\015\012",$text); 445f3f0262cSandi return htmlspecialchars($text); 446f3f0262cSandi} 447f3f0262cSandi 448f3f0262cSandi/** 44915fae107Sandi * Returns the specified local text in parsed format 45015fae107Sandi * 45115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 452f3f0262cSandi */ 453f3f0262cSandifunction parsedLocale($id){ 454f3f0262cSandi //disable section editing 455f3f0262cSandi global $parser; 456f3f0262cSandi $se = $parser['secedit']; 457f3f0262cSandi $parser['secedit'] = false; 458f3f0262cSandi //fetch parsed locale 459f3f0262cSandi $html = io_cacheParse(localeFN($id)); 460f3f0262cSandi //reset section editing 461f3f0262cSandi $parser['secedit'] = $se; 462f3f0262cSandi return $html; 463f3f0262cSandi} 464f3f0262cSandi 465f3f0262cSandi/** 46615fae107Sandi * Returns the specified local text in raw format 46715fae107Sandi * 46815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 469f3f0262cSandi */ 470f3f0262cSandifunction rawLocale($id){ 471f3f0262cSandi return io_readFile(localeFN($id)); 472f3f0262cSandi} 473f3f0262cSandi 474f3f0262cSandi 475f3f0262cSandi/** 47615fae107Sandi * Returns the parsed Wikitext for the given id and revision. 47715fae107Sandi * 47815fae107Sandi * If $excuse is true an explanation is returned if the file 47915fae107Sandi * wasn't found 48015fae107Sandi * 48115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 482f3f0262cSandi */ 483f3f0262cSandifunction parsedWiki($id,$rev='',$excuse=true){ 484f3f0262cSandi $file = wikiFN($id,$rev); 485f3f0262cSandi $ret = ''; 486f3f0262cSandi 487f3f0262cSandi //ensure $id is in global $ID (needed for parsing) 488f3f0262cSandi global $ID; 489f3f0262cSandi $ID = $id; 490f3f0262cSandi 491f3f0262cSandi if($rev){ 492f3f0262cSandi if(@file_exists($file)){ 493f3f0262cSandi $ret = parse(io_readFile($file)); 494f3f0262cSandi }elseif($excuse){ 495f3f0262cSandi $ret = parsedLocale('norev'); 496f3f0262cSandi } 497f3f0262cSandi }else{ 498f3f0262cSandi if(@file_exists($file)){ 499f3f0262cSandi $ret = io_cacheParse($file); 500f3f0262cSandi }elseif($excuse){ 501f3f0262cSandi $ret = parsedLocale('newpage'); 502f3f0262cSandi } 503f3f0262cSandi } 504f3f0262cSandi return $ret; 505f3f0262cSandi} 506f3f0262cSandi 507f3f0262cSandi/** 508f3f0262cSandi * Returns the raw WikiText 50915fae107Sandi * 51015fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 511f3f0262cSandi */ 512f3f0262cSandifunction rawWiki($id,$rev=''){ 513f3f0262cSandi return io_readFile(wikiFN($id,$rev)); 514f3f0262cSandi} 515f3f0262cSandi 516f3f0262cSandi/** 51715fae107Sandi * Returns the raw Wiki Text in three slices. 51815fae107Sandi * 51915fae107Sandi * The range parameter needs to have the form "from-to" 52015fae107Sandi * and gives the range of the section. 521f3f0262cSandi * The returned order is prefix, section and suffix. 52215fae107Sandi * 52315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 524f3f0262cSandi */ 525f3f0262cSandifunction rawWikiSlices($range,$id,$rev=''){ 526f3f0262cSandi list($from,$to) = split('-',$range,2); 527f3f0262cSandi $text = io_readFile(wikiFN($id,$rev)); 528f3f0262cSandi $text = split("\n",$text); 529f3f0262cSandi if(!$from) $from = 0; 530f3f0262cSandi if(!$to) $to = count($text); 531f3f0262cSandi 532f3f0262cSandi $slices[0] = join("\n",array_slice($text,0,$from)); 533f3f0262cSandi $slices[1] = join("\n",array_slice($text,$from,$to + 1 - $from)); 534f3f0262cSandi $slices[2] = join("\n",array_slice($text,$to+1)); 535f3f0262cSandi 536f3f0262cSandi return $slices; 537f3f0262cSandi} 538f3f0262cSandi 539f3f0262cSandi/** 54015fae107Sandi * Joins wiki text slices 54115fae107Sandi * 542f3f0262cSandi * function to join the text slices with correct lineendings again. 543f3f0262cSandi * When the pretty parameter is set to true it adds additional empty 544f3f0262cSandi * lines between sections if needed (used on saving). 54515fae107Sandi * 54615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 547f3f0262cSandi */ 548f3f0262cSandifunction con($pre,$text,$suf,$pretty=false){ 549f3f0262cSandi 550f3f0262cSandi if($pretty){ 551f3f0262cSandi if($pre && substr($pre,-1) != "\n") $pre .= "\n"; 552f3f0262cSandi if($suf && substr($text,-1) != "\n") $text .= "\n"; 553f3f0262cSandi } 554f3f0262cSandi 555f3f0262cSandi if($pre) $pre .= "\n"; 556f3f0262cSandi if($suf) $text .= "\n"; 557f3f0262cSandi return $pre.$text.$suf; 558f3f0262cSandi} 559f3f0262cSandi 560f3f0262cSandi/** 56115fae107Sandi * print debug messages 56215fae107Sandi * 563f3f0262cSandi * little function to print the content of a var 56415fae107Sandi * 56515fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 566f3f0262cSandi */ 567f3f0262cSandifunction dbg($msg,$hidden=false){ 568f3f0262cSandi (!$hidden) ? print '<pre class="dbg">' : print "<!--\n"; 569f3f0262cSandi print_r($msg); 570f3f0262cSandi (!$hidden) ? print '</pre>' : print "\n-->"; 571f3f0262cSandi} 572f3f0262cSandi 573f3f0262cSandi/** 574f3f0262cSandi * Add's an entry to the changelog 57515fae107Sandi * 57615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 577f3f0262cSandi */ 578652610a2Sandifunction addLogEntry($date,$id,$summary=""){ 579f3f0262cSandi global $conf; 580c1049928Sandi $id = cleanID($id);//FIXME not needed anymore? 581c1049928Sandi 582c1049928Sandi if(!@is_writable($conf['changelog'])){ 583c1049928Sandi msg($conf['changelog'].' is not writable!',-1); 584c1049928Sandi return; 585c1049928Sandi } 586c1049928Sandi 587652610a2Sandi if(!$date) $date = time(); //use current time if none supplied 588f3f0262cSandi $remote = $_SERVER['REMOTE_ADDR']; 589f3f0262cSandi $user = $_SERVER['REMOTE_USER']; 590f3f0262cSandi 591f3f0262cSandi $logline = join("\t",array($date,$remote,$id,$user,$summary))."\n"; 592f3f0262cSandi 593c1049928Sandi //FIXME: use adjusted io_saveFile instead 594f3f0262cSandi $fh = fopen($conf['changelog'],'a'); 595f3f0262cSandi if($fh){ 596f3f0262cSandi fwrite($fh,$logline); 597f3f0262cSandi fclose($fh); 598f3f0262cSandi } 599f3f0262cSandi} 600f3f0262cSandi 601f3f0262cSandi/** 602f3f0262cSandi * returns an array of recently changed files using the 603f3f0262cSandi * changelog 60415fae107Sandi * 60515fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 606f3f0262cSandi */ 607f3f0262cSandifunction getRecents($num=0,$incdel=false){ 608f3f0262cSandi global $conf; 609f3f0262cSandi $recent = array(); 610f3f0262cSandi if(!$num) $num = $conf['recent']; 611f3f0262cSandi 612c1049928Sandi if(!@is_readable($conf['changelog'])){ 613c1049928Sandi msg($conf['changelog'].' is not readable',-1); 614c1049928Sandi return $recent; 615c1049928Sandi } 616c1049928Sandi 617f3f0262cSandi $loglines = file($conf['changelog']); 618f3f0262cSandi rsort($loglines); //reverse sort on timestamp 619f3f0262cSandi 620f3f0262cSandi foreach ($loglines as $line){ 621f3f0262cSandi $line = rtrim($line); //remove newline 622f3f0262cSandi if(empty($line)) continue; //skip empty lines 623f3f0262cSandi $info = split("\t",$line); //split into parts 624f3f0262cSandi //add id if not in yet and file still exists and is allowed to read 625f3f0262cSandi if(!$recent[$info[2]] && 626f3f0262cSandi (@file_exists(wikiFN($info[2])) || $incdel) && 627f3f0262cSandi (auth_quickaclcheck($info[2]) >= AUTH_READ) 628f3f0262cSandi ){ 629f3f0262cSandi $recent[$info[2]]['date'] = $info[0]; 630f3f0262cSandi $recent[$info[2]]['ip'] = $info[1]; 631f3f0262cSandi $recent[$info[2]]['user'] = $info[3]; 632f3f0262cSandi $recent[$info[2]]['sum'] = $info[4]; 633f3f0262cSandi $recent[$info[2]]['del'] = !@file_exists(wikiFN($info[2])); 634f3f0262cSandi } 635f3f0262cSandi if(count($recent) >= $num){ 636f3f0262cSandi break; //finish if enough items found 637f3f0262cSandi } 638f3f0262cSandi } 639f3f0262cSandi return $recent; 640f3f0262cSandi} 641f3f0262cSandi 642f3f0262cSandi/** 643652610a2Sandi * gets additonal informations for a certain pagerevison 644652610a2Sandi * from the changelog 645652610a2Sandi * 646652610a2Sandi * @author Andreas Gohr <andi@splitbrain.org> 647652610a2Sandi */ 648652610a2Sandifunction getRevisionInfo($id,$rev){ 649652610a2Sandi global $conf; 650258641c6Sandi 651258641c6Sandi if(!$rev) return(null); 652258641c6Sandi 653c1049928Sandi $info = array(); 654c1049928Sandi if(!@is_readable($conf['changelog'])){ 655c1049928Sandi msg($conf['changelog'].' is not readable',-1); 656c1049928Sandi return $recent; 657c1049928Sandi } 658652610a2Sandi $loglines = file($conf['changelog']); 659652610a2Sandi $loglines = preg_grep("/$rev\t\d+\.\d+\.\d+\.\d+\t$id\t/",$loglines); 660652610a2Sandi rsort($loglines); //reverse sort on timestamp (shouldn't be needed) 661652610a2Sandi $line = split("\t",$loglines[0]); 662652610a2Sandi $info['date'] = $line[0]; 663652610a2Sandi $info['ip'] = $line[1]; 664652610a2Sandi $info['user'] = $line[3]; 665652610a2Sandi $info['sum'] = $line[4]; 666652610a2Sandi return $info; 667652610a2Sandi} 668652610a2Sandi 669652610a2Sandi/** 670f3f0262cSandi * Saves a wikitext by calling io_saveFile 67115fae107Sandi * 67215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 673f3f0262cSandi */ 674f3f0262cSandifunction saveWikiText($id,$text,$summary){ 675f3f0262cSandi global $conf; 676f3f0262cSandi global $lang; 677f3f0262cSandi umask($conf['umask']); 678f3f0262cSandi // ignore if no changes were made 679f3f0262cSandi if($text == rawWiki($id,'')){ 680f3f0262cSandi return; 681f3f0262cSandi } 682f3f0262cSandi 683f3f0262cSandi $file = wikiFN($id); 684f3f0262cSandi $old = saveOldRevision($id); 685f3f0262cSandi 686f3f0262cSandi if (empty($text)){ 687f3f0262cSandi // remove empty files 688f3f0262cSandi @unlink($file); 689f3f0262cSandi $del = true; 6903ce054b3Sandi //autoset summary on deletion 6913ce054b3Sandi if(empty($summary)) $summary = $lang['deleted']; 692f3f0262cSandi }else{ 693f3f0262cSandi // save file (datadir is created in io_saveFile) 694f3f0262cSandi io_saveFile($file,$text); 695f3f0262cSandi $del = false; 696f3f0262cSandi } 697f3f0262cSandi 698652610a2Sandi addLogEntry(@filemtime($file),$id,$summary); 699f3f0262cSandi notify($id,$old,$summary); 700f3f0262cSandi 701f3f0262cSandi //purge cache on add by updating the purgefile 702f3f0262cSandi if($conf['purgeonadd'] && (!$old || $del)){ 703f3f0262cSandi io_saveFile($conf['datadir'].'/.cache/purgefile',time()); 704f3f0262cSandi } 705f3f0262cSandi} 706f3f0262cSandi 707f3f0262cSandi/** 708f3f0262cSandi * moves the current version to the attic and returns its 709f3f0262cSandi * revision date 71015fae107Sandi * 71115fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 712f3f0262cSandi */ 713f3f0262cSandifunction saveOldRevision($id){ 714f3f0262cSandi global $conf; 715f3f0262cSandi umask($conf['umask']); 716f3f0262cSandi $oldf = wikiFN($id); 717f3f0262cSandi if(!@file_exists($oldf)) return ''; 718f3f0262cSandi $date = filemtime($oldf); 719f3f0262cSandi $newf = wikiFN($id,$date); 720f3f0262cSandi if(substr($newf,-3)=='.gz'){ 721f3f0262cSandi io_saveFile($newf,rawWiki($id)); 722f3f0262cSandi }else{ 723f3f0262cSandi io_makeFileDir($newf); 724f3f0262cSandi copy($oldf, $newf); 725f3f0262cSandi } 726f3f0262cSandi return $date; 727f3f0262cSandi} 728f3f0262cSandi 729f3f0262cSandi/** 730f3f0262cSandi * Sends a notify mail to the wikiadmin when a page was 731f3f0262cSandi * changed 73215fae107Sandi * 73315fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 734f3f0262cSandi */ 735f3f0262cSandifunction notify($id,$rev="",$summary=""){ 736f3f0262cSandi global $lang; 737f3f0262cSandi global $conf; 738f3f0262cSandi $hdrs =''; 739f3f0262cSandi if(empty($conf['notify'])) return; //notify enabled? 740f3f0262cSandi 741f3f0262cSandi $text = rawLocale('mailtext'); 742f3f0262cSandi $text = str_replace('@DATE@',date($conf['dformat']),$text); 743f3f0262cSandi $text = str_replace('@BROWSER@',$_SERVER['HTTP_USER_AGENT'],$text); 744f3f0262cSandi $text = str_replace('@IPADDRESS@',$_SERVER['REMOTE_ADDR'],$text); 745f3f0262cSandi $text = str_replace('@HOSTNAME@',gethostbyaddr($_SERVER['REMOTE_ADDR']),$text); 746ed7b5f09Sandi $text = str_replace('@NEWPAGE@',wl($id,'',true),$text); 747ed7b5f09Sandi $text = str_replace('@DOKUWIKIURL@',DOKU_URL,$text); 748f3f0262cSandi $text = str_replace('@SUMMARY@',$summary,$text); 7497a82afdcSandi $text = str_replace('@USER@',$_SERVER['REMOTE_USER'],$text); 750f3f0262cSandi 751f3f0262cSandi if($rev){ 752f3f0262cSandi $subject = $lang['mail_changed'].' '.$id; 753ed7b5f09Sandi $text = str_replace('@OLDPAGE@',wl($id,"rev=$rev",true),$text); 754f3f0262cSandi require_once("inc/DifferenceEngine.php"); 755f3f0262cSandi $df = new Diff(split("\n",rawWiki($id,$rev)), 756f3f0262cSandi split("\n",rawWiki($id))); 757f3f0262cSandi $dformat = new UnifiedDiffFormatter(); 758f3f0262cSandi $diff = $dformat->format($df); 759f3f0262cSandi }else{ 760f3f0262cSandi $subject=$lang['mail_newpage'].' '.$id; 761f3f0262cSandi $text = str_replace('@OLDPAGE@','none',$text); 762f3f0262cSandi $diff = rawWiki($id); 763f3f0262cSandi } 764f3f0262cSandi $text = str_replace('@DIFF@',$diff,$text); 765f3f0262cSandi 76644f669e9Sandi mail_send($conf['notify'],$subject,$text,$conf['mailfrom']); 767f3f0262cSandi} 768f3f0262cSandi 76915fae107Sandi/** 77015fae107Sandi * Return a list of available page revisons 77115fae107Sandi * 77215fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 77315fae107Sandi */ 774f3f0262cSandifunction getRevisions($id){ 775f3f0262cSandi $revd = dirname(wikiFN($id,'foo')); 776f3f0262cSandi $revs = array(); 777f3f0262cSandi $clid = cleanID($id); 778f3f0262cSandi if(strrpos($clid,':')) $clid = substr($clid,strrpos($clid,':')+1); //remove path 779f3f0262cSandi 780f3f0262cSandi if (is_dir($revd) && $dh = opendir($revd)) { 781f3f0262cSandi while (($file = readdir($dh)) !== false) { 782f3f0262cSandi if (is_dir($revd.'/'.$file)) continue; 783f3f0262cSandi if (preg_match('/^'.$clid.'\.(\d+)\.txt(\.gz)?$/',$file,$match)){ 784f3f0262cSandi $revs[]=$match[1]; 785f3f0262cSandi } 786f3f0262cSandi } 787f3f0262cSandi closedir($dh); 788f3f0262cSandi } 789f3f0262cSandi rsort($revs); 790f3f0262cSandi return $revs; 791f3f0262cSandi} 792f3f0262cSandi 793f3f0262cSandi/** 794f3f0262cSandi * downloads a file from the net and saves it to the given location 79515fae107Sandi * 79615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 797f3f0262cSandi */ 798f3f0262cSandifunction download($url,$file){ 799f3f0262cSandi $fp = @fopen($url,"rb"); 800f3f0262cSandi if(!$fp) return false; 801f3f0262cSandi 802f3f0262cSandi while(!feof($fp)){ 803f3f0262cSandi $cont.= fread($fp,1024); 804f3f0262cSandi } 805f3f0262cSandi fclose($fp); 806f3f0262cSandi 807f3f0262cSandi $fp2 = @fopen($file,"w"); 808f3f0262cSandi if(!$fp2) return false; 809f3f0262cSandi fwrite($fp2,$cont); 810f3f0262cSandi fclose($fp2); 811f3f0262cSandi return true; 812f3f0262cSandi} 813f3f0262cSandi 814f3f0262cSandi/** 815f3f0262cSandi * extracts the query from a google referer 81615fae107Sandi * 817*6b13307fSandi * @todo should be more generic and support yahoo et al 81815fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 819f3f0262cSandi */ 820f3f0262cSandifunction getGoogleQuery(){ 821f3f0262cSandi $url = parse_url($_SERVER['HTTP_REFERER']); 822f3f0262cSandi 823f3f0262cSandi if(!preg_match("#google\.#i",$url['host'])) return ''; 824f3f0262cSandi $query = array(); 825f3f0262cSandi parse_str($url['query'],$query); 826f3f0262cSandi 827f3f0262cSandi return $query['q']; 828f3f0262cSandi} 829f3f0262cSandi 830f3f0262cSandi/** 83115fae107Sandi * Try to set correct locale 83215fae107Sandi * 833095bfd5cSandi * @deprecated No longer used 83415fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 835f3f0262cSandi */ 836f3f0262cSandifunction setCorrectLocale(){ 837f3f0262cSandi global $conf; 838f3f0262cSandi global $lang; 839f3f0262cSandi 840f3f0262cSandi $enc = strtoupper($lang['encoding']); 841f3f0262cSandi foreach ($lang['locales'] as $loc){ 842f3f0262cSandi //try locale 843f3f0262cSandi if(@setlocale(LC_ALL,$loc)) return; 844f3f0262cSandi //try loceale with encoding 845f3f0262cSandi if(@setlocale(LC_ALL,"$loc.$enc")) return; 846f3f0262cSandi } 847f3f0262cSandi //still here? try to set from environment 848f3f0262cSandi @setlocale(LC_ALL,""); 849f3f0262cSandi} 850f3f0262cSandi 851f3f0262cSandi/** 852f3f0262cSandi * Return the human readable size of a file 853f3f0262cSandi * 854f3f0262cSandi * @param int $size A file size 855f3f0262cSandi * @param int $dec A number of decimal places 856f3f0262cSandi * @author Martin Benjamin <b.martin@cybernet.ch> 857f3f0262cSandi * @author Aidan Lister <aidan@php.net> 858f3f0262cSandi * @version 1.0.0 859f3f0262cSandi */ 860f31d5b73Sandifunction filesize_h($size, $dec = 1){ 861f3f0262cSandi $sizes = array('B', 'KB', 'MB', 'GB'); 862f3f0262cSandi $count = count($sizes); 863f3f0262cSandi $i = 0; 864f3f0262cSandi 865f3f0262cSandi while ($size >= 1024 && ($i < $count - 1)) { 866f3f0262cSandi $size /= 1024; 867f3f0262cSandi $i++; 868f3f0262cSandi } 869f3f0262cSandi 870f3f0262cSandi return round($size, $dec) . ' ' . $sizes[$i]; 871f3f0262cSandi} 872f3f0262cSandi 87315fae107Sandi/** 87415fae107Sandi * Run a few sanity checks 87515fae107Sandi * 87615fae107Sandi * @author Andreas Gohr <andi@splitbrain.org> 87715fae107Sandi */ 878f31d5b73Sandifunction getVersion(){ 879f31d5b73Sandi //import version string 880f31d5b73Sandi if(@file_exists('VERSION')){ 881f31d5b73Sandi //official release 882f31d5b73Sandi return 'Release '.io_readfile('VERSION'); 883f31d5b73Sandi }elseif(is_dir('_darcs')){ 884f31d5b73Sandi //darcs checkout 885f31d5b73Sandi $inv = file('_darcs/inventory'); 886f31d5b73Sandi $inv = preg_grep('#andi@splitbrain\.org\*\*\d{14}#',$inv); 887f31d5b73Sandi $cur = array_pop($inv); 888f31d5b73Sandi preg_match('#\*\*(\d{4})(\d{2})(\d{2})#',$cur,$matches); 889f31d5b73Sandi return 'Darcs '.$matches[1].'-'.$matches[2].'-'.$matches[3]; 890f31d5b73Sandi }else{ 891f31d5b73Sandi return 'snapshot?'; 892f31d5b73Sandi } 893f31d5b73Sandi} 894f31d5b73Sandi 895f31d5b73Sandi/** 896f31d5b73Sandi * Run a few sanity checks 897f31d5b73Sandi * 898f31d5b73Sandi * @author Andreas Gohr <andi@splitbrain.org> 899f31d5b73Sandi */ 900f3f0262cSandifunction check(){ 901f3f0262cSandi global $conf; 902f3f0262cSandi global $INFO; 903f3f0262cSandi 904f31d5b73Sandi msg('DokuWiki version: '.getVersion(),1); 905f31d5b73Sandi 90649022a38Sandi if(version_compare(phpversion(),'4.3.0','<')){ 90749022a38Sandi msg('Your PHP version is too old ('.phpversion().' vs. 4.3.+ recommended)',-1); 90849022a38Sandi }elseif(version_compare(phpversion(),'4.3.10','<')){ 90949022a38Sandi msg('Consider upgrading PHP to 4.3.10 or higher for security reasons (your version: '.phpversion().')',0); 91049022a38Sandi }else{ 91149022a38Sandi msg('PHP version '.phpversion(),1); 91249022a38Sandi } 91349022a38Sandi 914f3f0262cSandi if(is_writable($conf['changelog'])){ 915f3f0262cSandi msg('Changelog is writable',1); 916f3f0262cSandi }else{ 917f3f0262cSandi msg('Changelog is not writable',-1); 918f3f0262cSandi } 919f3f0262cSandi 920f3f0262cSandi if(is_writable($conf['datadir'])){ 921f3f0262cSandi msg('Datadir is writable',1); 922f3f0262cSandi }else{ 923f3f0262cSandi msg('Datadir is not writable',-1); 924f3f0262cSandi } 925f3f0262cSandi 926f3f0262cSandi if(is_writable($conf['olddir'])){ 927f3f0262cSandi msg('Attic is writable',1); 928f3f0262cSandi }else{ 929f3f0262cSandi msg('Attic is not writable',-1); 930f3f0262cSandi } 931f3f0262cSandi 932f3f0262cSandi if(is_writable($conf['mediadir'])){ 933f3f0262cSandi msg('Mediadir is writable',1); 934f3f0262cSandi }else{ 935f3f0262cSandi msg('Mediadir is not writable',-1); 936f3f0262cSandi } 937f3f0262cSandi 938f3f0262cSandi if(is_writable('conf/users.auth')){ 939f3f0262cSandi msg('conf/users.auth is writable',1); 940f3f0262cSandi }else{ 941f3f0262cSandi msg('conf/users.auth is not writable',0); 942f3f0262cSandi } 94393a9e835Sandi 94493a9e835Sandi if(function_exists('mb_strpos')){ 94593a9e835Sandi if(defined('UTF8_NOMBSTRING')){ 94693a9e835Sandi msg('mb_string extension is available but will not be used',0); 94793a9e835Sandi }else{ 94893a9e835Sandi msg('mb_string extension is available and will be used',1); 94993a9e835Sandi } 95093a9e835Sandi }else{ 95193a9e835Sandi msg('mb_string extension not available - PHP only replacements will be used',0); 95293a9e835Sandi } 953f3f0262cSandi 954f3f0262cSandi msg('Your current permission for this page is '.$INFO['perm'],0); 955f3f0262cSandi 956f3f0262cSandi if(is_writable($INFO['filepath'])){ 957f3f0262cSandi msg('The current page is writable by the webserver',0); 958f3f0262cSandi }else{ 959f3f0262cSandi msg('The current page is not writable by the webserver',0); 960f3f0262cSandi } 961f3f0262cSandi 962f3f0262cSandi if($INFO['writable']){ 963f3f0262cSandi msg('The current page is writable by you',0); 964f3f0262cSandi }else{ 965f3f0262cSandi msg('The current page is not writable you',0); 966f3f0262cSandi } 967f3f0262cSandi} 968f3f0262cSandi?> 969